scsi_transport_iscsi.c 49.6 KB
Newer Older
1
/*
L
Linus Torvalds 已提交
2 3 4
 * iSCSI transport class definitions
 *
 * Copyright (C) IBM Corporation, 2004
5 6 7
 * Copyright (C) Mike Christie, 2004 - 2005
 * Copyright (C) Dmitry Yusupov, 2004 - 2005
 * Copyright (C) Alex Aizman, 2004 - 2005
L
Linus Torvalds 已提交
8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
 */
#include <linux/module.h>
24
#include <linux/mutex.h>
25
#include <net/tcp.h>
L
Linus Torvalds 已提交
26 27 28 29 30
#include <scsi/scsi.h>
#include <scsi/scsi_host.h>
#include <scsi/scsi_device.h>
#include <scsi/scsi_transport.h>
#include <scsi/scsi_transport_iscsi.h>
31
#include <scsi/iscsi_if.h>
L
Linus Torvalds 已提交
32

33
#define ISCSI_SESSION_ATTRS 19
34
#define ISCSI_CONN_ATTRS 13
35
#define ISCSI_HOST_ATTRS 4
36
#define ISCSI_TRANSPORT_VERSION "2.0-869"
L
Linus Torvalds 已提交
37 38

struct iscsi_internal {
39
	int daemon_pid;
L
Linus Torvalds 已提交
40
	struct scsi_transport_template t;
41 42
	struct iscsi_transport *iscsi_transport;
	struct list_head list;
43
	struct device dev;
M
Mike Christie 已提交
44

45
	struct device_attribute *host_attrs[ISCSI_HOST_ATTRS + 1];
46
	struct transport_container conn_cont;
47
	struct device_attribute *conn_attrs[ISCSI_CONN_ATTRS + 1];
48
	struct transport_container session_cont;
49
	struct device_attribute *session_attrs[ISCSI_SESSION_ATTRS + 1];
L
Linus Torvalds 已提交
50 51
};

52
static atomic_t iscsi_session_nr; /* sysfs session id for next new session */
53
static struct workqueue_struct *iscsi_eh_timer_workq;
54

55 56 57
/*
 * list of registered transports and lock that must
 * be held while accessing list. The iscsi_transport_lock must
58
 * be acquired after the rx_queue_mutex.
59 60 61 62 63 64 65
 */
static LIST_HEAD(iscsi_transports);
static DEFINE_SPINLOCK(iscsi_transport_lock);

#define to_iscsi_internal(tmpl) \
	container_of(tmpl, struct iscsi_internal, t)

66 67
#define dev_to_iscsi_internal(_dev) \
	container_of(_dev, struct iscsi_internal, dev)
68

69
static void iscsi_transport_release(struct device *dev)
70
{
71
	struct iscsi_internal *priv = dev_to_iscsi_internal(dev);
72 73
	kfree(priv);
}
L
Linus Torvalds 已提交
74

75 76 77 78 79 80
/*
 * iscsi_transport_class represents the iscsi_transports that are
 * registered.
 */
static struct class iscsi_transport_class = {
	.name = "iscsi_transport",
81
	.dev_release = iscsi_transport_release,
82 83 84
};

static ssize_t
85 86
show_transport_handle(struct device *dev, struct device_attribute *attr,
		      char *buf)
87
{
88
	struct iscsi_internal *priv = dev_to_iscsi_internal(dev);
89
	return sprintf(buf, "%llu\n", (unsigned long long)iscsi_handle(priv->iscsi_transport));
90
}
91
static DEVICE_ATTR(handle, S_IRUGO, show_transport_handle, NULL);
92 93 94

#define show_transport_attr(name, format)				\
static ssize_t								\
95 96
show_transport_##name(struct device *dev, 				\
		      struct device_attribute *attr,char *buf)		\
97
{									\
98
	struct iscsi_internal *priv = dev_to_iscsi_internal(dev);	\
99 100
	return sprintf(buf, format"\n", priv->iscsi_transport->name);	\
}									\
101
static DEVICE_ATTR(name, S_IRUGO, show_transport_##name, NULL);
102 103 104 105

show_transport_attr(caps, "0x%x");

static struct attribute *iscsi_transport_attrs[] = {
106 107
	&dev_attr_handle.attr,
	&dev_attr_caps.attr,
108 109 110 111 112 113 114
	NULL,
};

static struct attribute_group iscsi_transport_group = {
	.attrs = iscsi_transport_attrs,
};

M
Mike Christie 已提交
115
static int iscsi_setup_host(struct transport_container *tc, struct device *dev,
116
			    struct device *cdev)
M
Mike Christie 已提交
117 118
{
	struct Scsi_Host *shost = dev_to_shost(dev);
119
	struct iscsi_cls_host *ihost = shost->shost_data;
M
Mike Christie 已提交
120 121 122 123

	memset(ihost, 0, sizeof(*ihost));
	INIT_LIST_HEAD(&ihost->sessions);
	mutex_init(&ihost->mutex);
124
	atomic_set(&ihost->nr_scans, 0);
M
Mike Christie 已提交
125

126
	snprintf(ihost->scan_workq_name, KOBJ_NAME_LEN, "iscsi_scan_%d",
M
Mike Christie 已提交
127
		shost->host_no);
128 129 130
	ihost->scan_workq = create_singlethread_workqueue(
						ihost->scan_workq_name);
	if (!ihost->scan_workq)
M
Mike Christie 已提交
131 132 133 134 135
		return -ENOMEM;
	return 0;
}

static int iscsi_remove_host(struct transport_container *tc, struct device *dev,
136
			     struct device *cdev)
M
Mike Christie 已提交
137 138
{
	struct Scsi_Host *shost = dev_to_shost(dev);
139
	struct iscsi_cls_host *ihost = shost->shost_data;
M
Mike Christie 已提交
140

141
	destroy_workqueue(ihost->scan_workq);
M
Mike Christie 已提交
142 143 144 145 146 147
	return 0;
}

static DECLARE_TRANSPORT_CLASS(iscsi_host_class,
			       "iscsi_host",
			       iscsi_setup_host,
M
Mike Christie 已提交
148
			       iscsi_remove_host,
M
Mike Christie 已提交
149 150
			       NULL);

151 152
static DECLARE_TRANSPORT_CLASS(iscsi_session_class,
			       "iscsi_session",
L
Linus Torvalds 已提交
153 154 155 156
			       NULL,
			       NULL,
			       NULL);

157 158
static DECLARE_TRANSPORT_CLASS(iscsi_connection_class,
			       "iscsi_connection",
L
Linus Torvalds 已提交
159 160 161
			       NULL,
			       NULL,
			       NULL);
162 163

static struct sock *nls;
164
static DEFINE_MUTEX(rx_queue_mutex);
165

166 167
static LIST_HEAD(sesslist);
static DEFINE_SPINLOCK(sesslock);
168 169
static LIST_HEAD(connlist);
static DEFINE_SPINLOCK(connlock);
170

171 172 173 174 175 176 177 178 179 180
static uint32_t iscsi_conn_get_sid(struct iscsi_cls_conn *conn)
{
	struct iscsi_cls_session *sess = iscsi_dev_to_session(conn->dev.parent);
	return sess->sid;
}

/*
 * Returns the matching session to a given sid
 */
static struct iscsi_cls_session *iscsi_session_lookup(uint32_t sid)
181 182 183 184 185 186
{
	unsigned long flags;
	struct iscsi_cls_session *sess;

	spin_lock_irqsave(&sesslock, flags);
	list_for_each_entry(sess, &sesslist, sess_list) {
187
		if (sess->sid == sid) {
188 189 190 191 192 193 194 195
			spin_unlock_irqrestore(&sesslock, flags);
			return sess;
		}
	}
	spin_unlock_irqrestore(&sesslock, flags);
	return NULL;
}

196 197 198 199
/*
 * Returns the matching connection to a given sid / cid tuple
 */
static struct iscsi_cls_conn *iscsi_conn_lookup(uint32_t sid, uint32_t cid)
200 201 202 203 204 205
{
	unsigned long flags;
	struct iscsi_cls_conn *conn;

	spin_lock_irqsave(&connlock, flags);
	list_for_each_entry(conn, &connlist, conn_list) {
206
		if ((conn->cid == cid) && (iscsi_conn_get_sid(conn) == sid)) {
207 208 209 210 211 212 213 214
			spin_unlock_irqrestore(&connlock, flags);
			return conn;
		}
	}
	spin_unlock_irqrestore(&connlock, flags);
	return NULL;
}

215 216 217 218
/*
 * The following functions can be used by LLDs that allocate
 * their own scsi_hosts or by software iscsi LLDs
 */
219 220 221 222 223 224 225 226 227
static struct {
	int value;
	char *name;
} iscsi_session_state_names[] = {
	{ ISCSI_SESSION_LOGGED_IN,	"LOGGED_IN" },
	{ ISCSI_SESSION_FAILED,		"FAILED" },
	{ ISCSI_SESSION_FREE,		"FREE" },
};

228
static const char *iscsi_session_state_name(int state)
229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266
{
	int i;
	char *name = NULL;

	for (i = 0; i < ARRAY_SIZE(iscsi_session_state_names); i++) {
		if (iscsi_session_state_names[i].value == state) {
			name = iscsi_session_state_names[i].name;
			break;
		}
	}
	return name;
}

int iscsi_session_chkready(struct iscsi_cls_session *session)
{
	unsigned long flags;
	int err;

	spin_lock_irqsave(&session->lock, flags);
	switch (session->state) {
	case ISCSI_SESSION_LOGGED_IN:
		err = 0;
		break;
	case ISCSI_SESSION_FAILED:
		err = DID_IMM_RETRY << 16;
		break;
	case ISCSI_SESSION_FREE:
		err = DID_NO_CONNECT << 16;
		break;
	default:
		err = DID_NO_CONNECT << 16;
		break;
	}
	spin_unlock_irqrestore(&session->lock, flags);
	return err;
}
EXPORT_SYMBOL_GPL(iscsi_session_chkready);

267 268 269 270
static void iscsi_session_release(struct device *dev)
{
	struct iscsi_cls_session *session = iscsi_dev_to_session(dev);
	struct Scsi_Host *shost;
271

272 273 274 275
	shost = iscsi_session_to_shost(session);
	scsi_host_put(shost);
	kfree(session);
}
276

277 278 279 280
static int iscsi_is_session_dev(const struct device *dev)
{
	return dev->release == iscsi_session_release;
}
281

282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299
static int iscsi_iter_session_fn(struct device *dev, void *data)
{
	void (* fn) (struct iscsi_cls_session *) = data;

	if (!iscsi_is_session_dev(dev))
		return 0;
	fn(iscsi_dev_to_session(dev));
	return 0;
}

void iscsi_host_for_each_session(struct Scsi_Host *shost,
				 void (*fn)(struct iscsi_cls_session *))
{
	device_for_each_child(&shost->shost_gendev, fn,
			      iscsi_iter_session_fn);
}
EXPORT_SYMBOL_GPL(iscsi_host_for_each_session);

300 301 302 303 304 305 306 307 308 309
/**
 * iscsi_scan_finished - helper to report when running scans are done
 * @shost: scsi host
 * @time: scan run time
 *
 * This function can be used by drives like qla4xxx to report to the scsi
 * layer when the scans it kicked off at module load time are done.
 */
int iscsi_scan_finished(struct Scsi_Host *shost, unsigned long time)
{
310
	struct iscsi_cls_host *ihost = shost->shost_data;
311 312 313 314 315 316 317 318
	/*
	 * qla4xxx will have kicked off some session unblocks before calling
	 * scsi_scan_host, so just wait for them to complete.
	 */
	return !atomic_read(&ihost->nr_scans);
}
EXPORT_SYMBOL_GPL(iscsi_scan_finished);

M
Mike Christie 已提交
319 320 321
static int iscsi_user_scan(struct Scsi_Host *shost, uint channel,
			   uint id, uint lun)
{
322
	struct iscsi_cls_host *ihost = shost->shost_data;
M
Mike Christie 已提交
323 324 325 326
	struct iscsi_cls_session *session;

	mutex_lock(&ihost->mutex);
	list_for_each_entry(session, &ihost->sessions, host_list) {
327
		if ((channel == SCAN_WILD_CARD || channel == 0) &&
M
Mike Christie 已提交
328
		    (id == SCAN_WILD_CARD || id == session->target_id))
329
			scsi_scan_target(&session->dev, 0,
M
Mike Christie 已提交
330 331 332 333 334 335 336
					 session->target_id, lun, 1);
	}
	mutex_unlock(&ihost->mutex);

	return 0;
}

337 338 339 340
static void iscsi_scan_session(struct work_struct *work)
{
	struct iscsi_cls_session *session =
			container_of(work, struct iscsi_cls_session, scan_work);
341
	struct Scsi_Host *shost = iscsi_session_to_shost(session);
342
	struct iscsi_cls_host *ihost = shost->shost_data;
343 344 345 346 347
	unsigned long flags;

	spin_lock_irqsave(&session->lock, flags);
	if (session->state != ISCSI_SESSION_LOGGED_IN) {
		spin_unlock_irqrestore(&session->lock, flags);
348
		goto done;
349 350 351 352 353
	}
	spin_unlock_irqrestore(&session->lock, flags);

	scsi_scan_target(&session->dev, 0, session->target_id,
			 SCAN_WILD_CARD, 1);
354 355
done:
	atomic_dec(&ihost->nr_scans);
356 357
}

D
David Howells 已提交
358
static void session_recovery_timedout(struct work_struct *work)
M
Mike Christie 已提交
359
{
D
David Howells 已提交
360 361 362
	struct iscsi_cls_session *session =
		container_of(work, struct iscsi_cls_session,
			     recovery_work.work);
363
	unsigned long flags;
M
Mike Christie 已提交
364

365 366 367
	iscsi_cls_session_printk(KERN_INFO, session,
				 "session recovery timed out after %d secs\n",
				 session->recovery_tmo);
M
Mike Christie 已提交
368

369 370 371 372 373 374 375 376 377 378 379 380 381
	spin_lock_irqsave(&session->lock, flags);
	switch (session->state) {
	case ISCSI_SESSION_FAILED:
		session->state = ISCSI_SESSION_FREE;
		break;
	case ISCSI_SESSION_LOGGED_IN:
	case ISCSI_SESSION_FREE:
		/* we raced with the unblock's flush */
		spin_unlock_irqrestore(&session->lock, flags);
		return;
	}
	spin_unlock_irqrestore(&session->lock, flags);

M
Mike Christie 已提交
382 383 384 385 386 387
	if (session->transport->session_recovery_timedout)
		session->transport->session_recovery_timedout(session);

	scsi_target_unblock(&session->dev);
}

388
static void __iscsi_unblock_session(struct work_struct *work)
389
{
390 391 392
	struct iscsi_cls_session *session =
			container_of(work, struct iscsi_cls_session,
				     unblock_work);
393
	struct Scsi_Host *shost = iscsi_session_to_shost(session);
394
	struct iscsi_cls_host *ihost = shost->shost_data;
395 396
	unsigned long flags;

397 398 399 400 401
	/*
	 * The recovery and unblock work get run from the same workqueue,
	 * so try to cancel it if it was going to run after this unblock.
	 */
	cancel_delayed_work(&session->recovery_work);
402 403 404
	spin_lock_irqsave(&session->lock, flags);
	session->state = ISCSI_SESSION_LOGGED_IN;
	spin_unlock_irqrestore(&session->lock, flags);
405 406
	/* start IO */
	scsi_target_unblock(&session->dev);
407 408 409 410 411 412 413 414 415
	/*
	 * Only do kernel scanning if the driver is properly hooked into
	 * the async scanning code (drivers like iscsi_tcp do login and
	 * scanning from userspace).
	 */
	if (shost->hostt->scan_finished) {
		if (queue_work(ihost->scan_workq, &session->scan_work))
			atomic_inc(&ihost->nr_scans);
	}
416
}
417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432

/**
 * iscsi_unblock_session - set a session as logged in and start IO.
 * @session: iscsi session
 *
 * Mark a session as ready to accept IO.
 */
void iscsi_unblock_session(struct iscsi_cls_session *session)
{
	queue_work(iscsi_eh_timer_workq, &session->unblock_work);
	/*
	 * make sure all the events have completed before tell the driver
	 * it is safe
	 */
	flush_workqueue(iscsi_eh_timer_workq);
}
M
Mike Christie 已提交
433 434
EXPORT_SYMBOL_GPL(iscsi_unblock_session);

435
static void __iscsi_block_session(struct work_struct *work)
M
Mike Christie 已提交
436
{
437 438 439
	struct iscsi_cls_session *session =
			container_of(work, struct iscsi_cls_session,
				     block_work);
440 441 442 443 444
	unsigned long flags;

	spin_lock_irqsave(&session->lock, flags);
	session->state = ISCSI_SESSION_FAILED;
	spin_unlock_irqrestore(&session->lock, flags);
M
Mike Christie 已提交
445
	scsi_target_block(&session->dev);
446 447
	queue_delayed_work(iscsi_eh_timer_workq, &session->recovery_work,
			   session->recovery_tmo * HZ);
M
Mike Christie 已提交
448
}
449 450 451 452 453

void iscsi_block_session(struct iscsi_cls_session *session)
{
	queue_work(iscsi_eh_timer_workq, &session->block_work);
}
M
Mike Christie 已提交
454 455
EXPORT_SYMBOL_GPL(iscsi_block_session);

M
Mike Christie 已提交
456 457 458 459 460 461
static void __iscsi_unbind_session(struct work_struct *work)
{
	struct iscsi_cls_session *session =
			container_of(work, struct iscsi_cls_session,
				     unbind_work);
	struct Scsi_Host *shost = iscsi_session_to_shost(session);
462
	struct iscsi_cls_host *ihost = shost->shost_data;
M
Mike Christie 已提交
463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479

	/* Prevent new scans and make sure scanning is not in progress */
	mutex_lock(&ihost->mutex);
	if (list_empty(&session->host_list)) {
		mutex_unlock(&ihost->mutex);
		return;
	}
	list_del_init(&session->host_list);
	mutex_unlock(&ihost->mutex);

	scsi_remove_target(&session->dev);
	iscsi_session_event(session, ISCSI_KEVENT_UNBIND_SESSION);
}

static int iscsi_unbind_session(struct iscsi_cls_session *session)
{
	struct Scsi_Host *shost = iscsi_session_to_shost(session);
480
	struct iscsi_cls_host *ihost = shost->shost_data;
M
Mike Christie 已提交
481

482
	return queue_work(ihost->scan_workq, &session->unbind_work);
M
Mike Christie 已提交
483 484
}

485
struct iscsi_cls_session *
486 487
iscsi_alloc_session(struct Scsi_Host *shost,
		    struct iscsi_transport *transport)
488 489 490
{
	struct iscsi_cls_session *session;

491 492
	session = kzalloc(sizeof(*session) + transport->sessiondata_size,
			  GFP_KERNEL);
493
	if (!session)
494 495
		return NULL;

496
	session->transport = transport;
M
Mike Christie 已提交
497
	session->recovery_tmo = 120;
498
	session->state = ISCSI_SESSION_FREE;
D
David Howells 已提交
499
	INIT_DELAYED_WORK(&session->recovery_work, session_recovery_timedout);
M
Mike Christie 已提交
500 501
	INIT_LIST_HEAD(&session->host_list);
	INIT_LIST_HEAD(&session->sess_list);
502 503
	INIT_WORK(&session->unblock_work, __iscsi_unblock_session);
	INIT_WORK(&session->block_work, __iscsi_block_session);
M
Mike Christie 已提交
504
	INIT_WORK(&session->unbind_work, __iscsi_unbind_session);
505
	INIT_WORK(&session->scan_work, iscsi_scan_session);
506
	spin_lock_init(&session->lock);
507

508 509
	/* this is released in the dev's release function */
	scsi_host_get(shost);
510 511 512
	session->dev.parent = &shost->shost_gendev;
	session->dev.release = iscsi_session_release;
	device_initialize(&session->dev);
513 514
	if (transport->sessiondata_size)
		session->dd_data = &session[1];
515 516 517 518
	return session;
}
EXPORT_SYMBOL_GPL(iscsi_alloc_session);

519
int iscsi_add_session(struct iscsi_cls_session *session, unsigned int target_id)
520 521
{
	struct Scsi_Host *shost = iscsi_session_to_shost(session);
522
	struct iscsi_cls_host *ihost;
M
Mike Christie 已提交
523
	unsigned long flags;
524
	int err;
525

M
Mike Christie 已提交
526
	ihost = shost->shost_data;
527
	session->sid = atomic_add_return(1, &iscsi_session_nr);
528
	session->target_id = target_id;
M
Mike Christie 已提交
529

530 531
	snprintf(session->dev.bus_id, BUS_ID_SIZE, "session%u",
		 session->sid);
532
	err = device_add(&session->dev);
533
	if (err) {
534 535
		iscsi_cls_session_printk(KERN_ERR, session,
					 "could not register session's dev\n");
536
		goto release_host;
537 538 539
	}
	transport_register_device(&session->dev);

M
Mike Christie 已提交
540 541 542 543
	spin_lock_irqsave(&sesslock, flags);
	list_add(&session->sess_list, &sesslist);
	spin_unlock_irqrestore(&sesslock, flags);

M
Mike Christie 已提交
544 545 546
	mutex_lock(&ihost->mutex);
	list_add(&session->host_list, &ihost->sessions);
	mutex_unlock(&ihost->mutex);
M
Mike Christie 已提交
547 548

	iscsi_session_event(session, ISCSI_KEVENT_CREATE_SESSION);
549
	return 0;
M
Mike Christie 已提交
550

551 552 553
release_host:
	scsi_host_put(shost);
	return err;
554
}
555
EXPORT_SYMBOL_GPL(iscsi_add_session);
556 557

/**
558 559 560
 * iscsi_create_session - create iscsi class session
 * @shost: scsi host
 * @transport: iscsi transport
561
 * @target_id: which target
562
 *
563
 * This can be called from a LLD or iscsi_transport.
564
 */
565 566
struct iscsi_cls_session *
iscsi_create_session(struct Scsi_Host *shost,
567 568
		     struct iscsi_transport *transport,
		     unsigned int target_id)
569 570 571 572 573 574 575
{
	struct iscsi_cls_session *session;

	session = iscsi_alloc_session(shost, transport);
	if (!session)
		return NULL;

576
	if (iscsi_add_session(session, target_id)) {
577 578 579 580 581 582 583
		iscsi_free_session(session);
		return NULL;
	}
	return session;
}
EXPORT_SYMBOL_GPL(iscsi_create_session);

M
Mike Christie 已提交
584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604
static void iscsi_conn_release(struct device *dev)
{
	struct iscsi_cls_conn *conn = iscsi_dev_to_conn(dev);
	struct device *parent = conn->dev.parent;

	kfree(conn);
	put_device(parent);
}

static int iscsi_is_conn_dev(const struct device *dev)
{
	return dev->release == iscsi_conn_release;
}

static int iscsi_iter_destroy_conn_fn(struct device *dev, void *data)
{
	if (!iscsi_is_conn_dev(dev))
		return 0;
	return iscsi_destroy_conn(iscsi_dev_to_conn(dev));
}

605
void iscsi_remove_session(struct iscsi_cls_session *session)
606
{
M
Mike Christie 已提交
607
	struct Scsi_Host *shost = iscsi_session_to_shost(session);
608
	struct iscsi_cls_host *ihost = shost->shost_data;
M
Mike Christie 已提交
609 610 611 612 613 614
	unsigned long flags;
	int err;

	spin_lock_irqsave(&sesslock, flags);
	list_del(&session->sess_list);
	spin_unlock_irqrestore(&sesslock, flags);
M
Mike Christie 已提交
615

616 617 618 619 620
	/* make sure there are no blocks/unblocks queued */
	flush_workqueue(iscsi_eh_timer_workq);
	/* make sure the timedout callout is not running */
	if (!cancel_delayed_work(&session->recovery_work))
		flush_workqueue(iscsi_eh_timer_workq);
M
Mike Christie 已提交
621 622 623
	/*
	 * If we are blocked let commands flow again. The lld or iscsi
	 * layer should set up the queuecommand to fail commands.
624 625
	 * We assume that LLD will not be calling block/unblock while
	 * removing the session.
M
Mike Christie 已提交
626
	 */
627 628 629
	spin_lock_irqsave(&session->lock, flags);
	session->state = ISCSI_SESSION_FREE;
	spin_unlock_irqrestore(&session->lock, flags);
630

631 632
	scsi_target_unblock(&session->dev);
	/* flush running scans then delete devices */
633
	flush_workqueue(ihost->scan_workq);
634
	__iscsi_unbind_session(&session->unbind_work);
M
Mike Christie 已提交
635

M
Mike Christie 已提交
636 637 638 639
	/* hw iscsi may not have removed all connections from session */
	err = device_for_each_child(&session->dev, NULL,
				    iscsi_iter_destroy_conn_fn);
	if (err)
640 641 642
		iscsi_cls_session_printk(KERN_ERR, session,
					 "Could not delete all connections "
					 "for session. Error %d.\n", err);
643

644
	transport_unregister_device(&session->dev);
645 646 647 648 649 650
	device_del(&session->dev);
}
EXPORT_SYMBOL_GPL(iscsi_remove_session);

void iscsi_free_session(struct iscsi_cls_session *session)
{
M
Mike Christie 已提交
651
	iscsi_session_event(session, ISCSI_KEVENT_DESTROY_SESSION);
652
	put_device(&session->dev);
653
}
654 655 656 657 658 659 660 661
EXPORT_SYMBOL_GPL(iscsi_free_session);

/**
 * iscsi_destroy_session - destroy iscsi session
 * @session: iscsi_session
 *
 * Can be called by a LLD or iscsi_transport. There must not be
 * any running connections.
662
 */
663 664 665 666 667 668
int iscsi_destroy_session(struct iscsi_cls_session *session)
{
	iscsi_remove_session(session);
	iscsi_free_session(session);
	return 0;
}
669 670 671 672 673 674 675 676 677 678
EXPORT_SYMBOL_GPL(iscsi_destroy_session);

/**
 * iscsi_create_conn - create iscsi class connection
 * @session: iscsi cls session
 * @cid: connection id
 *
 * This can be called from a LLD or iscsi_transport. The connection
 * is child of the session so cid must be unique for all connections
 * on the session.
679 680 681 682 683
 *
 * Since we do not support MCS, cid will normally be zero. In some cases
 * for software iscsi we could be trying to preallocate a connection struct
 * in which case there could be two connection structs and cid would be
 * non-zero.
684
 */
685 686 687 688 689
struct iscsi_cls_conn *
iscsi_create_conn(struct iscsi_cls_session *session, uint32_t cid)
{
	struct iscsi_transport *transport = session->transport;
	struct iscsi_cls_conn *conn;
M
Mike Christie 已提交
690
	unsigned long flags;
691 692 693 694 695 696 697 698 699 700 701
	int err;

	conn = kzalloc(sizeof(*conn) + transport->conndata_size, GFP_KERNEL);
	if (!conn)
		return NULL;

	if (transport->conndata_size)
		conn->dd_data = &conn[1];

	INIT_LIST_HEAD(&conn->conn_list);
	conn->transport = transport;
702
	conn->cid = cid;
703 704 705

	/* this is released in the dev's release function */
	if (!get_device(&session->dev))
706
		goto free_conn;
707

708
	snprintf(conn->dev.bus_id, BUS_ID_SIZE, "connection%d:%u",
709
		 session->sid, cid);
710 711 712 713
	conn->dev.parent = &session->dev;
	conn->dev.release = iscsi_conn_release;
	err = device_register(&conn->dev);
	if (err) {
714 715
		iscsi_cls_session_printk(KERN_ERR, session, "could not "
					 "register connection's dev\n");
716 717 718
		goto release_parent_ref;
	}
	transport_register_device(&conn->dev);
M
Mike Christie 已提交
719 720 721 722 723

	spin_lock_irqsave(&connlock, flags);
	list_add(&conn->conn_list, &connlist);
	conn->active = 1;
	spin_unlock_irqrestore(&connlock, flags);
724 725 726 727 728 729 730 731 732 733 734 735 736
	return conn;

release_parent_ref:
	put_device(&session->dev);
free_conn:
	kfree(conn);
	return NULL;
}

EXPORT_SYMBOL_GPL(iscsi_create_conn);

/**
 * iscsi_destroy_conn - destroy iscsi class connection
737
 * @conn: iscsi cls session
738
 *
M
Mike Christie 已提交
739
 * This can be called from a LLD or iscsi_transport.
740
 */
741 742
int iscsi_destroy_conn(struct iscsi_cls_conn *conn)
{
M
Mike Christie 已提交
743 744 745 746 747 748 749
	unsigned long flags;

	spin_lock_irqsave(&connlock, flags);
	conn->active = 0;
	list_del(&conn->conn_list);
	spin_unlock_irqrestore(&connlock, flags);

750 751 752 753 754 755 756 757 758
	transport_unregister_device(&conn->dev);
	device_unregister(&conn->dev);
	return 0;
}
EXPORT_SYMBOL_GPL(iscsi_destroy_conn);

/*
 * iscsi interface functions
 */
759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774
static struct iscsi_internal *
iscsi_if_transport_lookup(struct iscsi_transport *tt)
{
	struct iscsi_internal *priv;
	unsigned long flags;

	spin_lock_irqsave(&iscsi_transport_lock, flags);
	list_for_each_entry(priv, &iscsi_transports, list) {
		if (tt == priv->iscsi_transport) {
			spin_unlock_irqrestore(&iscsi_transport_lock, flags);
			return priv;
		}
	}
	spin_unlock_irqrestore(&iscsi_transport_lock, flags);
	return NULL;
}
L
Linus Torvalds 已提交
775

776
static int
777
iscsi_broadcast_skb(struct sk_buff *skb, gfp_t gfp)
778 779 780
{
	int rc;

781
	rc = netlink_broadcast(nls, skb, 0, 1, gfp);
782 783 784 785 786 787 788 789
	if (rc < 0) {
		printk(KERN_ERR "iscsi: can not broadcast skb (%d)\n", rc);
		return rc;
	}

	return 0;
}

790
static int
791
iscsi_unicast_skb(struct sk_buff *skb, int pid)
792 793
{
	int rc;
L
Linus Torvalds 已提交
794

795
	rc = netlink_unicast(nls, skb, pid, MSG_DONTWAIT);
796 797 798 799 800 801
	if (rc < 0) {
		printk(KERN_ERR "iscsi: can not unicast skb (%d)\n", rc);
		return rc;
	}

	return 0;
L
Linus Torvalds 已提交
802 803
}

804
int iscsi_recv_pdu(struct iscsi_cls_conn *conn, struct iscsi_hdr *hdr,
805
		   char *data, uint32_t data_size)
L
Linus Torvalds 已提交
806
{
807 808 809 810
	struct nlmsghdr	*nlh;
	struct sk_buff *skb;
	struct iscsi_uevent *ev;
	char *pdu;
811
	struct iscsi_internal *priv;
812 813 814
	int len = NLMSG_SPACE(sizeof(*ev) + sizeof(struct iscsi_hdr) +
			      data_size);

815 816 817 818
	priv = iscsi_if_transport_lookup(conn->transport);
	if (!priv)
		return -EINVAL;

819
	skb = alloc_skb(len, GFP_ATOMIC);
820
	if (!skb) {
821
		iscsi_conn_error(conn, ISCSI_ERR_CONN_FAILED);
822 823
		iscsi_cls_conn_printk(KERN_ERR, conn, "can not deliver "
				      "control PDU: OOM\n");
824 825
		return -ENOMEM;
	}
L
Linus Torvalds 已提交
826

827
	nlh = __nlmsg_put(skb, priv->daemon_pid, 0, 0, (len - sizeof(*nlh)), 0);
828 829 830 831
	ev = NLMSG_DATA(nlh);
	memset(ev, 0, sizeof(*ev));
	ev->transport_handle = iscsi_handle(conn->transport);
	ev->type = ISCSI_KEVENT_RECV_PDU;
832 833
	ev->r.recv_req.cid = conn->cid;
	ev->r.recv_req.sid = iscsi_conn_get_sid(conn);
834 835 836
	pdu = (char*)ev + sizeof(*ev);
	memcpy(pdu, hdr, sizeof(struct iscsi_hdr));
	memcpy(pdu + sizeof(struct iscsi_hdr), data, data_size);
L
Linus Torvalds 已提交
837

838
	return iscsi_unicast_skb(skb, priv->daemon_pid);
L
Linus Torvalds 已提交
839
}
840
EXPORT_SYMBOL_GPL(iscsi_recv_pdu);
L
Linus Torvalds 已提交
841

842
void iscsi_conn_error(struct iscsi_cls_conn *conn, enum iscsi_err error)
L
Linus Torvalds 已提交
843
{
844 845 846
	struct nlmsghdr	*nlh;
	struct sk_buff	*skb;
	struct iscsi_uevent *ev;
847
	struct iscsi_internal *priv;
848 849
	int len = NLMSG_SPACE(sizeof(*ev));

850 851 852 853
	priv = iscsi_if_transport_lookup(conn->transport);
	if (!priv)
		return;

854
	skb = alloc_skb(len, GFP_ATOMIC);
855
	if (!skb) {
856 857
		iscsi_cls_conn_printk(KERN_ERR, conn, "gracefully ignored "
				      "conn error (%d)\n", error);
858 859 860
		return;
	}

861
	nlh = __nlmsg_put(skb, priv->daemon_pid, 0, 0, (len - sizeof(*nlh)), 0);
862 863 864 865
	ev = NLMSG_DATA(nlh);
	ev->transport_handle = iscsi_handle(conn->transport);
	ev->type = ISCSI_KEVENT_CONN_ERROR;
	ev->r.connerror.error = error;
866 867
	ev->r.connerror.cid = conn->cid;
	ev->r.connerror.sid = iscsi_conn_get_sid(conn);
L
Linus Torvalds 已提交
868

869
	iscsi_broadcast_skb(skb, GFP_ATOMIC);
L
Linus Torvalds 已提交
870

871 872
	iscsi_cls_conn_printk(KERN_INFO, conn, "detected conn error (%d)\n",
			      error);
873 874 875 876 877 878 879 880 881 882 883 884 885
}
EXPORT_SYMBOL_GPL(iscsi_conn_error);

static int
iscsi_if_send_reply(int pid, int seq, int type, int done, int multi,
		      void *payload, int size)
{
	struct sk_buff	*skb;
	struct nlmsghdr	*nlh;
	int len = NLMSG_SPACE(size);
	int flags = multi ? NLM_F_MULTI : 0;
	int t = done ? NLMSG_DONE : type;

886
	skb = alloc_skb(len, GFP_ATOMIC);
887 888 889 890
	if (!skb) {
		printk(KERN_ERR "Could not allocate skb to send reply.\n");
		return -ENOMEM;
	}
891 892 893 894

	nlh = __nlmsg_put(skb, pid, seq, t, (len - sizeof(*nlh)), 0);
	nlh->nlmsg_flags = flags;
	memcpy(NLMSG_DATA(nlh), payload, size);
895
	return iscsi_unicast_skb(skb, pid);
L
Linus Torvalds 已提交
896 897
}

898
static int
899
iscsi_if_get_stats(struct iscsi_transport *transport, struct nlmsghdr *nlh)
900 901 902 903 904 905 906
{
	struct iscsi_uevent *ev = NLMSG_DATA(nlh);
	struct iscsi_stats *stats;
	struct sk_buff *skbstat;
	struct iscsi_cls_conn *conn;
	struct nlmsghdr	*nlhstat;
	struct iscsi_uevent *evstat;
907
	struct iscsi_internal *priv;
908 909 910 911 912
	int len = NLMSG_SPACE(sizeof(*ev) +
			      sizeof(struct iscsi_stats) +
			      sizeof(struct iscsi_stats_custom) *
			      ISCSI_STATS_CUSTOM_MAX);
	int err = 0;
913

914 915 916 917
	priv = iscsi_if_transport_lookup(transport);
	if (!priv)
		return -EINVAL;

918
	conn = iscsi_conn_lookup(ev->u.get_stats.sid, ev->u.get_stats.cid);
919 920
	if (!conn)
		return -EEXIST;
921

922 923
	do {
		int actual_size;
924

925
		skbstat = alloc_skb(len, GFP_ATOMIC);
926
		if (!skbstat) {
927 928
			iscsi_cls_conn_printk(KERN_ERR, conn, "can not "
					      "deliver stats: OOM\n");
929 930 931
			return -ENOMEM;
		}

932
		nlhstat = __nlmsg_put(skbstat, priv->daemon_pid, 0, 0,
933 934 935 936 937
				      (len - sizeof(*nlhstat)), 0);
		evstat = NLMSG_DATA(nlhstat);
		memset(evstat, 0, sizeof(*evstat));
		evstat->transport_handle = iscsi_handle(conn->transport);
		evstat->type = nlh->nlmsg_type;
938 939 940 941
		evstat->u.get_stats.cid =
			ev->u.get_stats.cid;
		evstat->u.get_stats.sid =
			ev->u.get_stats.sid;
942 943 944 945
		stats = (struct iscsi_stats *)
			((char*)evstat + sizeof(*evstat));
		memset(stats, 0, sizeof(*stats));

946
		transport->get_stats(conn, stats);
947 948 949 950 951 952
		actual_size = NLMSG_SPACE(sizeof(struct iscsi_uevent) +
					  sizeof(struct iscsi_stats) +
					  sizeof(struct iscsi_stats_custom) *
					  stats->custom_length);
		actual_size -= sizeof(*nlhstat);
		actual_size = NLMSG_LENGTH(actual_size);
953
		skb_trim(skbstat, NLMSG_ALIGN(actual_size));
954 955
		nlhstat->nlmsg_len = actual_size;

956
		err = iscsi_unicast_skb(skbstat, priv->daemon_pid);
957 958 959
	} while (err < 0 && err != -ECONNREFUSED);

	return err;
960 961
}

962
/**
M
Mike Christie 已提交
963 964 965
 * iscsi_session_event - send session destr. completion event
 * @session: iscsi class session
 * @event: type of event
966
 */
M
Mike Christie 已提交
967 968
int iscsi_session_event(struct iscsi_cls_session *session,
			enum iscsi_uevent_e event)
969 970 971 972 973 974 975 976
{
	struct iscsi_internal *priv;
	struct Scsi_Host *shost;
	struct iscsi_uevent *ev;
	struct sk_buff  *skb;
	struct nlmsghdr *nlh;
	int rc, len = NLMSG_SPACE(sizeof(*ev));

M
Mike Christie 已提交
977
	priv = iscsi_if_transport_lookup(session->transport);
978 979 980 981
	if (!priv)
		return -EINVAL;
	shost = iscsi_session_to_shost(session);

982
	skb = alloc_skb(len, GFP_KERNEL);
983
	if (!skb) {
984 985 986
		iscsi_cls_session_printk(KERN_ERR, session,
					 "Cannot notify userspace of session "
					 "event %u\n", event);
987 988 989 990 991
		return -ENOMEM;
	}

	nlh = __nlmsg_put(skb, priv->daemon_pid, 0, 0, (len - sizeof(*nlh)), 0);
	ev = NLMSG_DATA(nlh);
M
Mike Christie 已提交
992
	ev->transport_handle = iscsi_handle(session->transport);
993

M
Mike Christie 已提交
994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008
	ev->type = event;
	switch (event) {
	case ISCSI_KEVENT_DESTROY_SESSION:
		ev->r.d_session.host_no = shost->host_no;
		ev->r.d_session.sid = session->sid;
		break;
	case ISCSI_KEVENT_CREATE_SESSION:
		ev->r.c_session_ret.host_no = shost->host_no;
		ev->r.c_session_ret.sid = session->sid;
		break;
	case ISCSI_KEVENT_UNBIND_SESSION:
		ev->r.unbind_session.host_no = shost->host_no;
		ev->r.unbind_session.sid = session->sid;
		break;
	default:
1009 1010
		iscsi_cls_session_printk(KERN_ERR, session, "Invalid event "
					 "%u.\n", event);
M
Mike Christie 已提交
1011
		kfree_skb(skb);
1012 1013 1014 1015 1016 1017 1018
		return -EINVAL;
	}

	/*
	 * this will occur if the daemon is not up, so we just warn
	 * the user and when the daemon is restarted it will handle it
	 */
1019
	rc = iscsi_broadcast_skb(skb, GFP_KERNEL);
1020
	if (rc < 0)
1021 1022 1023 1024
		iscsi_cls_session_printk(KERN_ERR, session,
					 "Cannot notify userspace of session "
					 "event %u. Check iscsi daemon\n",
					 event);
1025 1026
	return rc;
}
M
Mike Christie 已提交
1027
EXPORT_SYMBOL_GPL(iscsi_session_event);
1028

1029
static int
1030 1031 1032
iscsi_if_create_session(struct iscsi_internal *priv, struct iscsi_uevent *ev,
			uint32_t host_no, uint32_t initial_cmdsn,
			uint16_t cmds_max, uint16_t queue_depth)
1033 1034
{
	struct iscsi_transport *transport = priv->iscsi_transport;
1035
	struct iscsi_cls_session *session;
1036
	struct Scsi_Host *shost = NULL;
1037

1038 1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052
	/*
	 * Software iscsi allocates a host per session, but
	 * offload drivers (and possibly iser one day) allocate a host per
	 * hba/nic/rnic. Offload will match a host here, but software will
	 * return a new hostno after the create_session callback has returned.
	 */
	if (host_no != UINT_MAX) {
		shost = scsi_host_lookup(host_no);
		if (IS_ERR(shost)) {
			printk(KERN_ERR "Could not find host no %u to "
			       "create session\n", host_no);
			return -ENODEV;
		}
	}

1053
	session = transport->create_session(shost, cmds_max, queue_depth,
1054 1055 1056
					    initial_cmdsn, &host_no);
	if (shost)
		scsi_host_put(shost);
1057
	if (!session)
1058
		return -ENOMEM;
1059

1060
	ev->r.c_session_ret.host_no = host_no;
1061
	ev->r.c_session_ret.sid = session->sid;
1062 1063 1064 1065
	return 0;
}

static int
1066
iscsi_if_create_conn(struct iscsi_transport *transport, struct iscsi_uevent *ev)
1067
{
1068
	struct iscsi_cls_conn *conn;
1069
	struct iscsi_cls_session *session;
1070

1071 1072
	session = iscsi_session_lookup(ev->u.c_conn.sid);
	if (!session) {
1073
		printk(KERN_ERR "iscsi: invalid session %d.\n",
1074
		       ev->u.c_conn.sid);
1075
		return -EINVAL;
1076
	}
1077

1078
	conn = transport->create_conn(session, ev->u.c_conn.cid);
1079
	if (!conn) {
1080 1081
		iscsi_cls_session_printk(KERN_ERR, session,
					 "couldn't create a new connection.");
1082
		return -ENOMEM;
1083
	}
1084

1085 1086
	ev->r.c_conn_ret.sid = session->sid;
	ev->r.c_conn_ret.cid = conn->cid;
1087
	return 0;
L
Linus Torvalds 已提交
1088 1089
}

1090 1091 1092
static int
iscsi_if_destroy_conn(struct iscsi_transport *transport, struct iscsi_uevent *ev)
{
1093
	struct iscsi_cls_conn *conn;
1094

1095
	conn = iscsi_conn_lookup(ev->u.d_conn.sid, ev->u.d_conn.cid);
1096
	if (!conn)
1097
		return -EINVAL;
1098

1099 1100 1101
	if (transport->destroy_conn)
		transport->destroy_conn(conn);
	return 0;
1102 1103
}

1104 1105 1106 1107 1108 1109
static int
iscsi_set_param(struct iscsi_transport *transport, struct iscsi_uevent *ev)
{
	char *data = (char*)ev + sizeof(*ev);
	struct iscsi_cls_conn *conn;
	struct iscsi_cls_session *session;
1110
	int err = 0, value = 0;
1111 1112 1113 1114 1115 1116 1117

	session = iscsi_session_lookup(ev->u.set_param.sid);
	conn = iscsi_conn_lookup(ev->u.set_param.sid, ev->u.set_param.cid);
	if (!conn || !session)
		return -EINVAL;

	switch (ev->u.set_param.param) {
M
Mike Christie 已提交
1118
	case ISCSI_PARAM_SESS_RECOVERY_TMO:
1119
		sscanf(data, "%d", &value);
M
Mike Christie 已提交
1120 1121 1122
		if (value != 0)
			session->recovery_tmo = value;
		break;
1123
	default:
1124 1125
		err = transport->set_param(conn, ev->u.set_param.param,
					   data, ev->u.set_param.len);
1126 1127 1128 1129 1130
	}

	return err;
}

1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164
static int
iscsi_if_transport_ep(struct iscsi_transport *transport,
		      struct iscsi_uevent *ev, int msg_type)
{
	struct sockaddr *dst_addr;
	int rc = 0;

	switch (msg_type) {
	case ISCSI_UEVENT_TRANSPORT_EP_CONNECT:
		if (!transport->ep_connect)
			return -EINVAL;

		dst_addr = (struct sockaddr *)((char*)ev + sizeof(*ev));
		rc = transport->ep_connect(dst_addr,
					   ev->u.ep_connect.non_blocking,
					   &ev->r.ep_connect_ret.handle);
		break;
	case ISCSI_UEVENT_TRANSPORT_EP_POLL:
		if (!transport->ep_poll)
			return -EINVAL;

		ev->r.retcode = transport->ep_poll(ev->u.ep_poll.ep_handle,
						   ev->u.ep_poll.timeout_ms);
		break;
	case ISCSI_UEVENT_TRANSPORT_EP_DISCONNECT:
		if (!transport->ep_disconnect)
			return -EINVAL;

		transport->ep_disconnect(ev->u.ep_disconnect.ep_handle);
		break;
	}
	return rc;
}

1165 1166 1167 1168
static int
iscsi_tgt_dscvr(struct iscsi_transport *transport,
		struct iscsi_uevent *ev)
{
1169
	struct Scsi_Host *shost;
1170
	struct sockaddr *dst_addr;
1171
	int err;
1172 1173 1174 1175

	if (!transport->tgt_dscvr)
		return -EINVAL;

1176 1177 1178 1179 1180 1181 1182 1183
	shost = scsi_host_lookup(ev->u.tgt_dscvr.host_no);
	if (IS_ERR(shost)) {
		printk(KERN_ERR "target discovery could not find host no %u\n",
		       ev->u.tgt_dscvr.host_no);
		return -ENODEV;
	}


1184
	dst_addr = (struct sockaddr *)((char*)ev + sizeof(*ev));
1185 1186 1187 1188
	err = transport->tgt_dscvr(shost, ev->u.tgt_dscvr.type,
				   ev->u.tgt_dscvr.enable, dst_addr);
	scsi_host_put(shost);
	return err;
1189 1190
}

1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214
static int
iscsi_set_host_param(struct iscsi_transport *transport,
		     struct iscsi_uevent *ev)
{
	char *data = (char*)ev + sizeof(*ev);
	struct Scsi_Host *shost;
	int err;

	if (!transport->set_host_param)
		return -ENOSYS;

	shost = scsi_host_lookup(ev->u.set_host_param.host_no);
	if (IS_ERR(shost)) {
		printk(KERN_ERR "set_host_param could not find host no %u\n",
		       ev->u.set_host_param.host_no);
		return -ENODEV;
	}

	err = transport->set_host_param(shost, ev->u.set_host_param.param,
					data, ev->u.set_host_param.len);
	scsi_host_put(shost);
	return err;
}

1215 1216 1217 1218
static int
iscsi_if_recv_msg(struct sk_buff *skb, struct nlmsghdr *nlh)
{
	int err = 0;
1219
	uint32_t host_no = UINT_MAX;
1220 1221 1222
	struct iscsi_uevent *ev = NLMSG_DATA(nlh);
	struct iscsi_transport *transport = NULL;
	struct iscsi_internal *priv;
1223 1224
	struct iscsi_cls_session *session;
	struct iscsi_cls_conn *conn;
1225 1226 1227 1228 1229 1230

	priv = iscsi_if_transport_lookup(iscsi_ptr(ev->transport_handle));
	if (!priv)
		return -EINVAL;
	transport = priv->iscsi_transport;

1231 1232 1233
	if (!try_module_get(transport->owner))
		return -EINVAL;

1234 1235
	priv->daemon_pid = NETLINK_CREDS(skb)->pid;

1236 1237
	switch (nlh->nlmsg_type) {
	case ISCSI_UEVENT_CREATE_SESSION:
1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248
		err = iscsi_if_create_session(priv, ev, host_no,
					      ev->u.c_session.initial_cmdsn,
					      ev->u.c_session.cmds_max,
					      ev->u.c_session.queue_depth);
		break;
	case ISCSI_UEVENT_CREATE_BOUND_SESSION:
		err = iscsi_if_create_session(priv, ev,
					ev->u.c_bound_session.host_no,
					ev->u.c_bound_session.initial_cmdsn,
					ev->u.c_bound_session.cmds_max,
					ev->u.c_bound_session.queue_depth);
1249 1250
		break;
	case ISCSI_UEVENT_DESTROY_SESSION:
1251
		session = iscsi_session_lookup(ev->u.d_session.sid);
M
Mike Christie 已提交
1252
		if (session)
1253
			transport->destroy_session(session);
M
Mike Christie 已提交
1254 1255 1256 1257 1258 1259 1260 1261
		else
			err = -EINVAL;
		break;
	case ISCSI_UEVENT_UNBIND_SESSION:
		session = iscsi_session_lookup(ev->u.d_session.sid);
		if (session)
			iscsi_unbind_session(session);
		else
1262
			err = -EINVAL;
1263 1264 1265 1266 1267 1268 1269 1270
		break;
	case ISCSI_UEVENT_CREATE_CONN:
		err = iscsi_if_create_conn(transport, ev);
		break;
	case ISCSI_UEVENT_DESTROY_CONN:
		err = iscsi_if_destroy_conn(transport, ev);
		break;
	case ISCSI_UEVENT_BIND_CONN:
1271 1272
		session = iscsi_session_lookup(ev->u.b_conn.sid);
		conn = iscsi_conn_lookup(ev->u.b_conn.sid, ev->u.b_conn.cid);
1273 1274 1275

		if (session && conn)
			ev->r.retcode =	transport->bind_conn(session, conn,
1276
					ev->u.b_conn.transport_eph,
1277 1278 1279
					ev->u.b_conn.is_leading);
		else
			err = -EINVAL;
1280 1281
		break;
	case ISCSI_UEVENT_SET_PARAM:
1282
		err = iscsi_set_param(transport, ev);
1283 1284
		break;
	case ISCSI_UEVENT_START_CONN:
1285
		conn = iscsi_conn_lookup(ev->u.start_conn.sid, ev->u.start_conn.cid);
1286 1287 1288 1289
		if (conn)
			ev->r.retcode = transport->start_conn(conn);
		else
			err = -EINVAL;
1290 1291
		break;
	case ISCSI_UEVENT_STOP_CONN:
1292
		conn = iscsi_conn_lookup(ev->u.stop_conn.sid, ev->u.stop_conn.cid);
1293 1294 1295 1296
		if (conn)
			transport->stop_conn(conn, ev->u.stop_conn.flag);
		else
			err = -EINVAL;
1297 1298
		break;
	case ISCSI_UEVENT_SEND_PDU:
1299
		conn = iscsi_conn_lookup(ev->u.send_pdu.sid, ev->u.send_pdu.cid);
1300 1301 1302 1303 1304 1305 1306
		if (conn)
			ev->r.retcode =	transport->send_pdu(conn,
				(struct iscsi_hdr*)((char*)ev + sizeof(*ev)),
				(char*)ev + sizeof(*ev) + ev->u.send_pdu.hdr_size,
				ev->u.send_pdu.data_size);
		else
			err = -EINVAL;
1307 1308
		break;
	case ISCSI_UEVENT_GET_STATS:
1309
		err = iscsi_if_get_stats(transport, nlh);
1310
		break;
1311 1312 1313 1314 1315
	case ISCSI_UEVENT_TRANSPORT_EP_CONNECT:
	case ISCSI_UEVENT_TRANSPORT_EP_POLL:
	case ISCSI_UEVENT_TRANSPORT_EP_DISCONNECT:
		err = iscsi_if_transport_ep(transport, ev, nlh->nlmsg_type);
		break;
1316 1317 1318
	case ISCSI_UEVENT_TGT_DSCVR:
		err = iscsi_tgt_dscvr(transport, ev);
		break;
1319 1320 1321
	case ISCSI_UEVENT_SET_HOST_PARAM:
		err = iscsi_set_host_param(transport, ev);
		break;
1322
	default:
1323
		err = -ENOSYS;
1324 1325 1326
		break;
	}

1327
	module_put(transport->owner);
1328 1329 1330
	return err;
}

1331
/*
1332 1333
 * Get message from skb.  Each message is processed by iscsi_if_recv_msg.
 * Malformed skbs with wrong lengths or invalid creds are not processed.
1334
 */
1335
static void
1336
iscsi_if_rx(struct sk_buff *skb)
1337
{
1338
	mutex_lock(&rx_queue_mutex);
1339 1340 1341 1342 1343 1344 1345 1346 1347 1348
	while (skb->len >= NLMSG_SPACE(0)) {
		int err;
		uint32_t rlen;
		struct nlmsghdr	*nlh;
		struct iscsi_uevent *ev;

		nlh = nlmsg_hdr(skb);
		if (nlh->nlmsg_len < sizeof(*nlh) ||
		    skb->len < nlh->nlmsg_len) {
			break;
1349 1350
		}

1351 1352 1353 1354
		ev = NLMSG_DATA(nlh);
		rlen = NLMSG_ALIGN(nlh->nlmsg_len);
		if (rlen > skb->len)
			rlen = skb->len;
1355

1356 1357 1358 1359
		err = iscsi_if_recv_msg(skb, nlh);
		if (err) {
			ev->type = ISCSI_KEVENT_IF_ERROR;
			ev->iferror = err;
1360
		}
1361 1362 1363 1364 1365 1366 1367 1368 1369 1370 1371 1372 1373 1374
		do {
			/*
			 * special case for GET_STATS:
			 * on success - sending reply and stats from
			 * inside of if_recv_msg(),
			 * on error - fall through.
			 */
			if (ev->type == ISCSI_UEVENT_GET_STATS && !err)
				break;
			err = iscsi_if_send_reply(
				NETLINK_CREDS(skb)->pid, nlh->nlmsg_seq,
				nlh->nlmsg_type, 0, 0, ev, sizeof(*ev));
		} while (err < 0 && err != -ECONNREFUSED);
		skb_pull(skb, rlen);
1375
	}
1376
	mutex_unlock(&rx_queue_mutex);
1377
}
L
Linus Torvalds 已提交
1378

1379
#define ISCSI_CLASS_ATTR(_prefix,_name,_mode,_show,_store)		\
1380
struct device_attribute dev_attr_##_prefix##_##_name =	\
1381 1382
	__ATTR(_name,_mode,_show,_store)

L
Linus Torvalds 已提交
1383
/*
1384
 * iSCSI connection attrs
L
Linus Torvalds 已提交
1385
 */
1386
#define iscsi_conn_attr_show(param)					\
1387
static ssize_t								\
1388 1389
show_conn_param_##param(struct device *dev, 				\
			struct device_attribute *attr, char *buf)	\
1390
{									\
1391
	struct iscsi_cls_conn *conn = iscsi_dev_to_conn(dev->parent);	\
1392
	struct iscsi_transport *t = conn->transport;			\
1393
	return t->get_conn_param(conn, param, buf);			\
1394 1395
}

1396 1397 1398
#define iscsi_conn_attr(field, param)					\
	iscsi_conn_attr_show(param)					\
static ISCSI_CLASS_ATTR(conn, field, S_IRUGO, show_conn_param_##param,	\
1399
			NULL);
1400

1401 1402 1403 1404 1405 1406 1407 1408 1409 1410 1411
iscsi_conn_attr(max_recv_dlength, ISCSI_PARAM_MAX_RECV_DLENGTH);
iscsi_conn_attr(max_xmit_dlength, ISCSI_PARAM_MAX_XMIT_DLENGTH);
iscsi_conn_attr(header_digest, ISCSI_PARAM_HDRDGST_EN);
iscsi_conn_attr(data_digest, ISCSI_PARAM_DATADGST_EN);
iscsi_conn_attr(ifmarker, ISCSI_PARAM_IFMARKER_EN);
iscsi_conn_attr(ofmarker, ISCSI_PARAM_OFMARKER_EN);
iscsi_conn_attr(persistent_port, ISCSI_PARAM_PERSISTENT_PORT);
iscsi_conn_attr(port, ISCSI_PARAM_CONN_PORT);
iscsi_conn_attr(exp_statsn, ISCSI_PARAM_EXP_STATSN);
iscsi_conn_attr(persistent_address, ISCSI_PARAM_PERSISTENT_ADDRESS);
iscsi_conn_attr(address, ISCSI_PARAM_CONN_ADDRESS);
1412 1413
iscsi_conn_attr(ping_tmo, ISCSI_PARAM_PING_TMO);
iscsi_conn_attr(recv_tmo, ISCSI_PARAM_RECV_TMO);
L
Linus Torvalds 已提交
1414 1415

/*
1416
 * iSCSI session attrs
L
Linus Torvalds 已提交
1417
 */
1418
#define iscsi_session_attr_show(param, perm)				\
1419
static ssize_t								\
1420 1421
show_session_param_##param(struct device *dev,				\
			   struct device_attribute *attr, char *buf)	\
1422
{									\
1423 1424
	struct iscsi_cls_session *session = 				\
		iscsi_dev_to_session(dev->parent);			\
1425
	struct iscsi_transport *t = session->transport;			\
1426 1427 1428
									\
	if (perm && !capable(CAP_SYS_ADMIN))				\
		return -EACCES;						\
1429
	return t->get_session_param(session, param, buf);		\
1430 1431
}

1432 1433
#define iscsi_session_attr(field, param, perm)				\
	iscsi_session_attr_show(param, perm)				\
1434
static ISCSI_CLASS_ATTR(sess, field, S_IRUGO, show_session_param_##param, \
1435 1436
			NULL);

1437 1438 1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450
iscsi_session_attr(targetname, ISCSI_PARAM_TARGET_NAME, 0);
iscsi_session_attr(initial_r2t, ISCSI_PARAM_INITIAL_R2T_EN, 0);
iscsi_session_attr(max_outstanding_r2t, ISCSI_PARAM_MAX_R2T, 0);
iscsi_session_attr(immediate_data, ISCSI_PARAM_IMM_DATA_EN, 0);
iscsi_session_attr(first_burst_len, ISCSI_PARAM_FIRST_BURST, 0);
iscsi_session_attr(max_burst_len, ISCSI_PARAM_MAX_BURST, 0);
iscsi_session_attr(data_pdu_in_order, ISCSI_PARAM_PDU_INORDER_EN, 0);
iscsi_session_attr(data_seq_in_order, ISCSI_PARAM_DATASEQ_INORDER_EN, 0);
iscsi_session_attr(erl, ISCSI_PARAM_ERL, 0);
iscsi_session_attr(tpgt, ISCSI_PARAM_TPGT, 0);
iscsi_session_attr(username, ISCSI_PARAM_USERNAME, 1);
iscsi_session_attr(username_in, ISCSI_PARAM_USERNAME_IN, 1);
iscsi_session_attr(password, ISCSI_PARAM_PASSWORD, 1);
iscsi_session_attr(password_in, ISCSI_PARAM_PASSWORD_IN, 1);
1451 1452 1453
iscsi_session_attr(fast_abort, ISCSI_PARAM_FAST_ABORT, 0);
iscsi_session_attr(abort_tmo, ISCSI_PARAM_ABORT_TMO, 0);
iscsi_session_attr(lu_reset_tmo, ISCSI_PARAM_LU_RESET_TMO, 0);
L
Linus Torvalds 已提交
1454

1455
static ssize_t
1456 1457
show_priv_session_state(struct device *dev, struct device_attribute *attr,
			char *buf)
1458
{
1459
	struct iscsi_cls_session *session = iscsi_dev_to_session(dev->parent);
1460 1461 1462 1463 1464
	return sprintf(buf, "%s\n", iscsi_session_state_name(session->state));
}
static ISCSI_CLASS_ATTR(priv_sess, state, S_IRUGO, show_priv_session_state,
			NULL);

1465 1466
#define iscsi_priv_session_attr_show(field, format)			\
static ssize_t								\
1467 1468
show_priv_session_##field(struct device *dev, 				\
			  struct device_attribute *attr, char *buf)	\
1469
{									\
1470 1471
	struct iscsi_cls_session *session = 				\
			iscsi_dev_to_session(dev->parent);		\
1472 1473 1474 1475 1476 1477 1478
	return sprintf(buf, format"\n", session->field);		\
}

#define iscsi_priv_session_attr(field, format)				\
	iscsi_priv_session_attr_show(field, format)			\
static ISCSI_CLASS_ATTR(priv_sess, field, S_IRUGO, show_priv_session_##field, \
			NULL)
M
Mike Christie 已提交
1479
iscsi_priv_session_attr(recovery_tmo, "%d");
1480

1481 1482 1483 1484 1485
/*
 * iSCSI host attrs
 */
#define iscsi_host_attr_show(param)					\
static ssize_t								\
1486 1487
show_host_param_##param(struct device *dev, 				\
			struct device_attribute *attr, char *buf)	\
1488
{									\
1489
	struct Scsi_Host *shost = transport_class_to_shost(dev);	\
1490 1491 1492 1493 1494 1495 1496 1497 1498
	struct iscsi_internal *priv = to_iscsi_internal(shost->transportt); \
	return priv->iscsi_transport->get_host_param(shost, param, buf); \
}

#define iscsi_host_attr(field, param)					\
	iscsi_host_attr_show(param)					\
static ISCSI_CLASS_ATTR(host, field, S_IRUGO, show_host_param_##param,	\
			NULL);

1499
iscsi_host_attr(netdev, ISCSI_HOST_PARAM_NETDEV_NAME);
1500
iscsi_host_attr(hwaddress, ISCSI_HOST_PARAM_HWADDRESS);
1501
iscsi_host_attr(ipaddress, ISCSI_HOST_PARAM_IPADDRESS);
1502
iscsi_host_attr(initiatorname, ISCSI_HOST_PARAM_INITIATOR_NAME);
1503

1504 1505
#define SETUP_PRIV_SESSION_RD_ATTR(field)				\
do {									\
1506
	priv->session_attrs[count] = &dev_attr_priv_sess_##field; \
1507 1508 1509
	count++;							\
} while (0)

1510

1511 1512 1513
#define SETUP_SESSION_RD_ATTR(field, param_flag)			\
do {									\
	if (tt->param_mask & param_flag) {				\
1514
		priv->session_attrs[count] = &dev_attr_sess_##field; \
L
Linus Torvalds 已提交
1515
		count++;						\
1516 1517 1518 1519 1520 1521
	}								\
} while (0)

#define SETUP_CONN_RD_ATTR(field, param_flag)				\
do {									\
	if (tt->param_mask & param_flag) {				\
1522
		priv->conn_attrs[count] = &dev_attr_conn_##field; \
1523 1524 1525
		count++;						\
	}								\
} while (0)
L
Linus Torvalds 已提交
1526

1527 1528 1529
#define SETUP_HOST_RD_ATTR(field, param_flag)				\
do {									\
	if (tt->host_param_mask & param_flag) {				\
1530
		priv->host_attrs[count] = &dev_attr_host_##field; \
1531 1532 1533 1534
		count++;						\
	}								\
} while (0)

1535 1536
static int iscsi_session_match(struct attribute_container *cont,
			   struct device *dev)
L
Linus Torvalds 已提交
1537
{
1538
	struct iscsi_cls_session *session;
L
Linus Torvalds 已提交
1539
	struct Scsi_Host *shost;
1540 1541 1542 1543
	struct iscsi_internal *priv;

	if (!iscsi_is_session_dev(dev))
		return 0;
L
Linus Torvalds 已提交
1544

1545 1546
	session = iscsi_dev_to_session(dev);
	shost = iscsi_session_to_shost(session);
1547
	if (!shost->transportt)
L
Linus Torvalds 已提交
1548 1549
		return 0;

1550 1551
	priv = to_iscsi_internal(shost->transportt);
	if (priv->session_cont.ac.class != &iscsi_session_class.class)
L
Linus Torvalds 已提交
1552 1553
		return 0;

1554
	return &priv->session_cont.ac == cont;
L
Linus Torvalds 已提交
1555 1556
}

1557 1558 1559
static int iscsi_conn_match(struct attribute_container *cont,
			   struct device *dev)
{
1560 1561
	struct iscsi_cls_session *session;
	struct iscsi_cls_conn *conn;
L
Linus Torvalds 已提交
1562
	struct Scsi_Host *shost;
1563
	struct iscsi_internal *priv;
L
Linus Torvalds 已提交
1564

1565
	if (!iscsi_is_conn_dev(dev))
L
Linus Torvalds 已提交
1566 1567
		return 0;

1568 1569 1570 1571
	conn = iscsi_dev_to_conn(dev);
	session = iscsi_dev_to_session(conn->dev.parent);
	shost = iscsi_session_to_shost(session);

1572
	if (!shost->transportt)
L
Linus Torvalds 已提交
1573 1574
		return 0;

1575 1576 1577
	priv = to_iscsi_internal(shost->transportt);
	if (priv->conn_cont.ac.class != &iscsi_connection_class.class)
		return 0;
L
Linus Torvalds 已提交
1578

1579 1580 1581
	return &priv->conn_cont.ac == cont;
}

M
Mike Christie 已提交
1582 1583 1584 1585 1586 1587 1588 1589 1590 1591 1592 1593 1594 1595 1596 1597 1598 1599
static int iscsi_host_match(struct attribute_container *cont,
			    struct device *dev)
{
	struct Scsi_Host *shost;
	struct iscsi_internal *priv;

	if (!scsi_is_host_device(dev))
		return 0;

	shost = dev_to_shost(dev);
	if (!shost->transportt  ||
	    shost->transportt->host_attrs.ac.class != &iscsi_host_class.class)
		return 0;

        priv = to_iscsi_internal(shost->transportt);
        return &priv->t.host_attrs.ac == cont;
}

1600 1601
struct scsi_transport_template *
iscsi_register_transport(struct iscsi_transport *tt)
1602 1603 1604 1605 1606 1607 1608 1609 1610
{
	struct iscsi_internal *priv;
	unsigned long flags;
	int count = 0, err;

	BUG_ON(!tt);

	priv = iscsi_if_transport_lookup(tt);
	if (priv)
1611
		return NULL;
1612

J
Jes Sorensen 已提交
1613
	priv = kzalloc(sizeof(*priv), GFP_KERNEL);
1614
	if (!priv)
1615
		return NULL;
1616
	INIT_LIST_HEAD(&priv->list);
1617
	priv->daemon_pid = -1;
1618
	priv->iscsi_transport = tt;
M
Mike Christie 已提交
1619
	priv->t.user_scan = iscsi_user_scan;
1620 1621
	if (!(tt->caps & CAP_DATA_PATH_OFFLOAD))
		priv->t.create_work_queue = 1;
1622

1623 1624 1625
	priv->dev.class = &iscsi_transport_class;
	snprintf(priv->dev.bus_id, BUS_ID_SIZE, "%s", tt->name);
	err = device_register(&priv->dev);
1626 1627 1628
	if (err)
		goto free_priv;

1629
	err = sysfs_create_group(&priv->dev.kobj, &iscsi_transport_group);
1630
	if (err)
1631
		goto unregister_dev;
1632

M
Mike Christie 已提交
1633 1634 1635 1636
	/* host parameters */
	priv->t.host_attrs.ac.attrs = &priv->host_attrs[0];
	priv->t.host_attrs.ac.class = &iscsi_host_class.class;
	priv->t.host_attrs.ac.match = iscsi_host_match;
1637
	priv->t.host_size = sizeof(struct iscsi_cls_host);
M
Mike Christie 已提交
1638 1639
	transport_container_register(&priv->t.host_attrs);

1640
	SETUP_HOST_RD_ATTR(netdev, ISCSI_HOST_NETDEV_NAME);
1641
	SETUP_HOST_RD_ATTR(ipaddress, ISCSI_HOST_IPADDRESS);
1642
	SETUP_HOST_RD_ATTR(hwaddress, ISCSI_HOST_HWADDRESS);
1643
	SETUP_HOST_RD_ATTR(initiatorname, ISCSI_HOST_INITIATOR_NAME);
1644 1645 1646 1647
	BUG_ON(count > ISCSI_HOST_ATTRS);
	priv->host_attrs[count] = NULL;
	count = 0;

1648 1649 1650 1651 1652
	/* connection parameters */
	priv->conn_cont.ac.attrs = &priv->conn_attrs[0];
	priv->conn_cont.ac.class = &iscsi_connection_class.class;
	priv->conn_cont.ac.match = iscsi_conn_match;
	transport_container_register(&priv->conn_cont);
L
Linus Torvalds 已提交
1653

1654 1655 1656 1657 1658 1659 1660 1661
	SETUP_CONN_RD_ATTR(max_recv_dlength, ISCSI_MAX_RECV_DLENGTH);
	SETUP_CONN_RD_ATTR(max_xmit_dlength, ISCSI_MAX_XMIT_DLENGTH);
	SETUP_CONN_RD_ATTR(header_digest, ISCSI_HDRDGST_EN);
	SETUP_CONN_RD_ATTR(data_digest, ISCSI_DATADGST_EN);
	SETUP_CONN_RD_ATTR(ifmarker, ISCSI_IFMARKER_EN);
	SETUP_CONN_RD_ATTR(ofmarker, ISCSI_OFMARKER_EN);
	SETUP_CONN_RD_ATTR(address, ISCSI_CONN_ADDRESS);
	SETUP_CONN_RD_ATTR(port, ISCSI_CONN_PORT);
1662
	SETUP_CONN_RD_ATTR(exp_statsn, ISCSI_EXP_STATSN);
1663 1664
	SETUP_CONN_RD_ATTR(persistent_address, ISCSI_PERSISTENT_ADDRESS);
	SETUP_CONN_RD_ATTR(persistent_port, ISCSI_PERSISTENT_PORT);
1665 1666
	SETUP_CONN_RD_ATTR(ping_tmo, ISCSI_PING_TMO);
	SETUP_CONN_RD_ATTR(recv_tmo, ISCSI_RECV_TMO);
1667 1668 1669

	BUG_ON(count > ISCSI_CONN_ATTRS);
	priv->conn_attrs[count] = NULL;
L
Linus Torvalds 已提交
1670 1671
	count = 0;

1672 1673 1674 1675 1676 1677
	/* session parameters */
	priv->session_cont.ac.attrs = &priv->session_attrs[0];
	priv->session_cont.ac.class = &iscsi_session_class.class;
	priv->session_cont.ac.match = iscsi_session_match;
	transport_container_register(&priv->session_cont);

1678 1679 1680 1681 1682 1683 1684 1685
	SETUP_SESSION_RD_ATTR(initial_r2t, ISCSI_INITIAL_R2T_EN);
	SETUP_SESSION_RD_ATTR(max_outstanding_r2t, ISCSI_MAX_R2T);
	SETUP_SESSION_RD_ATTR(immediate_data, ISCSI_IMM_DATA_EN);
	SETUP_SESSION_RD_ATTR(first_burst_len, ISCSI_FIRST_BURST);
	SETUP_SESSION_RD_ATTR(max_burst_len, ISCSI_MAX_BURST);
	SETUP_SESSION_RD_ATTR(data_pdu_in_order, ISCSI_PDU_INORDER_EN);
	SETUP_SESSION_RD_ATTR(data_seq_in_order, ISCSI_DATASEQ_INORDER_EN);
	SETUP_SESSION_RD_ATTR(erl, ISCSI_ERL);
1686 1687
	SETUP_SESSION_RD_ATTR(targetname, ISCSI_TARGET_NAME);
	SETUP_SESSION_RD_ATTR(tpgt, ISCSI_TPGT);
1688 1689 1690 1691
	SETUP_SESSION_RD_ATTR(password, ISCSI_USERNAME);
	SETUP_SESSION_RD_ATTR(password_in, ISCSI_USERNAME_IN);
	SETUP_SESSION_RD_ATTR(username, ISCSI_PASSWORD);
	SETUP_SESSION_RD_ATTR(username_in, ISCSI_PASSWORD_IN);
1692
	SETUP_SESSION_RD_ATTR(fast_abort, ISCSI_FAST_ABORT);
1693 1694
	SETUP_SESSION_RD_ATTR(abort_tmo, ISCSI_ABORT_TMO);
	SETUP_SESSION_RD_ATTR(lu_reset_tmo,ISCSI_LU_RESET_TMO);
M
Mike Christie 已提交
1695
	SETUP_PRIV_SESSION_RD_ATTR(recovery_tmo);
1696
	SETUP_PRIV_SESSION_RD_ATTR(state);
1697

1698 1699 1700 1701 1702 1703 1704 1705
	BUG_ON(count > ISCSI_SESSION_ATTRS);
	priv->session_attrs[count] = NULL;

	spin_lock_irqsave(&iscsi_transport_lock, flags);
	list_add(&priv->list, &iscsi_transports);
	spin_unlock_irqrestore(&iscsi_transport_lock, flags);

	printk(KERN_NOTICE "iscsi: registered transport (%s)\n", tt->name);
1706
	return &priv->t;
L
Linus Torvalds 已提交
1707

1708 1709
unregister_dev:
	device_unregister(&priv->dev);
1710 1711
free_priv:
	kfree(priv);
1712
	return NULL;
L
Linus Torvalds 已提交
1713
}
1714 1715 1716 1717 1718 1719 1720 1721 1722
EXPORT_SYMBOL_GPL(iscsi_register_transport);

int iscsi_unregister_transport(struct iscsi_transport *tt)
{
	struct iscsi_internal *priv;
	unsigned long flags;

	BUG_ON(!tt);

1723
	mutex_lock(&rx_queue_mutex);
1724 1725 1726 1727 1728 1729 1730 1731 1732 1733

	priv = iscsi_if_transport_lookup(tt);
	BUG_ON (!priv);

	spin_lock_irqsave(&iscsi_transport_lock, flags);
	list_del(&priv->list);
	spin_unlock_irqrestore(&iscsi_transport_lock, flags);

	transport_container_unregister(&priv->conn_cont);
	transport_container_unregister(&priv->session_cont);
M
Mike Christie 已提交
1734
	transport_container_unregister(&priv->t.host_attrs);
1735

1736 1737
	sysfs_remove_group(&priv->dev.kobj, &iscsi_transport_group);
	device_unregister(&priv->dev);
1738
	mutex_unlock(&rx_queue_mutex);
L
Linus Torvalds 已提交
1739

1740 1741 1742
	return 0;
}
EXPORT_SYMBOL_GPL(iscsi_unregister_transport);
L
Linus Torvalds 已提交
1743 1744 1745

static __init int iscsi_transport_init(void)
{
1746
	int err;
L
Linus Torvalds 已提交
1747

M
Meelis Roos 已提交
1748
	printk(KERN_INFO "Loading iSCSI transport class v%s.\n",
1749 1750
		ISCSI_TRANSPORT_VERSION);

1751 1752
	atomic_set(&iscsi_session_nr, 0);

1753
	err = class_register(&iscsi_transport_class);
L
Linus Torvalds 已提交
1754 1755
	if (err)
		return err;
1756

M
Mike Christie 已提交
1757
	err = transport_class_register(&iscsi_host_class);
1758 1759 1760
	if (err)
		goto unregister_transport_class;

M
Mike Christie 已提交
1761 1762 1763 1764
	err = transport_class_register(&iscsi_connection_class);
	if (err)
		goto unregister_host_class;

1765 1766 1767 1768
	err = transport_class_register(&iscsi_session_class);
	if (err)
		goto unregister_conn_class;

1769
	nls = netlink_kernel_create(&init_net, NETLINK_ISCSI, 1, iscsi_if_rx, NULL,
1770
			THIS_MODULE);
1771 1772
	if (!nls) {
		err = -ENOBUFS;
1773
		goto unregister_session_class;
1774 1775
	}

1776 1777 1778 1779
	iscsi_eh_timer_workq = create_singlethread_workqueue("iscsi_eh");
	if (!iscsi_eh_timer_workq)
		goto release_nls;

1780
	return 0;
1781

1782
release_nls:
1783
	netlink_kernel_release(nls);
1784 1785 1786 1787
unregister_session_class:
	transport_class_unregister(&iscsi_session_class);
unregister_conn_class:
	transport_class_unregister(&iscsi_connection_class);
M
Mike Christie 已提交
1788 1789
unregister_host_class:
	transport_class_unregister(&iscsi_host_class);
1790 1791 1792
unregister_transport_class:
	class_unregister(&iscsi_transport_class);
	return err;
L
Linus Torvalds 已提交
1793 1794 1795 1796
}

static void __exit iscsi_transport_exit(void)
{
1797
	destroy_workqueue(iscsi_eh_timer_workq);
1798
	netlink_kernel_release(nls);
1799 1800
	transport_class_unregister(&iscsi_connection_class);
	transport_class_unregister(&iscsi_session_class);
M
Mike Christie 已提交
1801
	transport_class_unregister(&iscsi_host_class);
1802
	class_unregister(&iscsi_transport_class);
L
Linus Torvalds 已提交
1803 1804 1805 1806 1807
}

module_init(iscsi_transport_init);
module_exit(iscsi_transport_exit);

1808 1809 1810 1811
MODULE_AUTHOR("Mike Christie <michaelc@cs.wisc.edu>, "
	      "Dmitry Yusupov <dmitry_yus@yahoo.com>, "
	      "Alex Aizman <itn780@yahoo.com>");
MODULE_DESCRIPTION("iSCSI Transport Interface");
L
Linus Torvalds 已提交
1812
MODULE_LICENSE("GPL");
1813
MODULE_VERSION(ISCSI_TRANSPORT_VERSION);