css.c 26.0 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
2
 * driver for channel subsystem
L
Linus Torvalds 已提交
3
 *
4 5 6 7
 * Copyright IBM Corp. 2002, 2009
 *
 * Author(s): Arnd Bergmann (arndb@de.ibm.com)
 *	      Cornelia Huck (cornelia.huck@de.ibm.com)
L
Linus Torvalds 已提交
8
 */
9 10 11 12

#define KMSG_COMPONENT "cio"
#define pr_fmt(fmt) KMSG_COMPONENT ": " fmt

L
Linus Torvalds 已提交
13 14 15 16 17 18
#include <linux/module.h>
#include <linux/init.h>
#include <linux/device.h>
#include <linux/slab.h>
#include <linux/errno.h>
#include <linux/list.h>
19
#include <linux/reboot.h>
20
#include <linux/suspend.h>
21
#include <asm/isc.h>
22
#include <asm/crw.h>
L
Linus Torvalds 已提交
23 24 25 26 27 28

#include "css.h"
#include "cio.h"
#include "cio_debug.h"
#include "ioasm.h"
#include "chsc.h"
29
#include "device.h"
30
#include "idset.h"
31
#include "chp.h"
L
Linus Torvalds 已提交
32 33

int css_init_done = 0;
34
int max_ssid;
L
Linus Torvalds 已提交
35

36
struct channel_subsystem *channel_subsystems[__MAX_CSSID + 1];
L
Linus Torvalds 已提交
37

38
int
39 40 41 42 43 44 45 46
for_each_subchannel(int(*fn)(struct subchannel_id, void *), void *data)
{
	struct subchannel_id schid;
	int ret;

	init_subchannel_id(&schid);
	ret = -ENODEV;
	do {
47 48 49 50 51 52 53
		do {
			ret = fn(schid, data);
			if (ret)
				break;
		} while (schid.sch_no++ < __MAX_SUBCHANNEL);
		schid.sch_no = 0;
	} while (schid.ssid++ < max_ssid);
54 55 56
	return ret;
}

57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85
struct cb_data {
	void *data;
	struct idset *set;
	int (*fn_known_sch)(struct subchannel *, void *);
	int (*fn_unknown_sch)(struct subchannel_id, void *);
};

static int call_fn_known_sch(struct device *dev, void *data)
{
	struct subchannel *sch = to_subchannel(dev);
	struct cb_data *cb = data;
	int rc = 0;

	idset_sch_del(cb->set, sch->schid);
	if (cb->fn_known_sch)
		rc = cb->fn_known_sch(sch, cb->data);
	return rc;
}

static int call_fn_unknown_sch(struct subchannel_id schid, void *data)
{
	struct cb_data *cb = data;
	int rc = 0;

	if (idset_sch_contains(cb->set, schid))
		rc = cb->fn_unknown_sch(schid, cb->data);
	return rc;
}

86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104
static int call_fn_all_sch(struct subchannel_id schid, void *data)
{
	struct cb_data *cb = data;
	struct subchannel *sch;
	int rc = 0;

	sch = get_subchannel_by_schid(schid);
	if (sch) {
		if (cb->fn_known_sch)
			rc = cb->fn_known_sch(sch, cb->data);
		put_device(&sch->dev);
	} else {
		if (cb->fn_unknown_sch)
			rc = cb->fn_unknown_sch(schid, cb->data);
	}

	return rc;
}

105 106 107 108 109 110 111 112 113 114
int for_each_subchannel_staged(int (*fn_known)(struct subchannel *, void *),
			       int (*fn_unknown)(struct subchannel_id,
			       void *), void *data)
{
	struct cb_data cb;
	int rc;

	cb.data = data;
	cb.fn_known_sch = fn_known;
	cb.fn_unknown_sch = fn_unknown;
115 116 117 118 119 120 121 122

	cb.set = idset_sch_new();
	if (!cb.set)
		/* fall back to brute force scanning in case of oom */
		return for_each_subchannel(call_fn_all_sch, &cb);

	idset_fill(cb.set);

123 124 125 126 127 128 129 130 131 132 133 134 135
	/* Process registered subchannels. */
	rc = bus_for_each_dev(&css_bus_type, NULL, &cb, call_fn_known_sch);
	if (rc)
		goto out;
	/* Process unregistered subchannels. */
	if (fn_unknown)
		rc = for_each_subchannel(call_fn_unknown_sch, &cb);
out:
	idset_free(cb.set);

	return rc;
}

L
Linus Torvalds 已提交
136
static struct subchannel *
137
css_alloc_subchannel(struct subchannel_id schid)
L
Linus Torvalds 已提交
138 139 140 141 142 143 144
{
	struct subchannel *sch;
	int ret;

	sch = kmalloc (sizeof (*sch), GFP_KERNEL | GFP_DMA);
	if (sch == NULL)
		return ERR_PTR(-ENOMEM);
145
	ret = cio_validate_subchannel (sch, schid);
L
Linus Torvalds 已提交
146 147 148 149 150 151 152 153 154 155 156 157 158
	if (ret < 0) {
		kfree(sch);
		return ERR_PTR(ret);
	}
	return sch;
}

static void
css_subchannel_release(struct device *dev)
{
	struct subchannel *sch;

	sch = to_subchannel(dev);
C
Cornelia Huck 已提交
159
	if (!cio_is_console(sch->schid)) {
160 161 162
		/* Reset intparm to zeroes. */
		sch->config.intparm = 0;
		cio_commit_config(sch);
C
Cornelia Huck 已提交
163
		kfree(sch->lock);
L
Linus Torvalds 已提交
164
		kfree(sch);
C
Cornelia Huck 已提交
165
	}
L
Linus Torvalds 已提交
166 167
}

168
static int css_sch_device_register(struct subchannel *sch)
169 170 171 172
{
	int ret;

	mutex_lock(&sch->reg_mutex);
173 174
	dev_set_name(&sch->dev, "0.%x.%04x", sch->schid.ssid,
		     sch->schid.sch_no);
175 176 177 178 179
	ret = device_register(&sch->dev);
	mutex_unlock(&sch->reg_mutex);
	return ret;
}

180 181 182 183
/**
 * css_sch_device_unregister - unregister a subchannel
 * @sch: subchannel to be unregistered
 */
184 185 186
void css_sch_device_unregister(struct subchannel *sch)
{
	mutex_lock(&sch->reg_mutex);
187 188
	if (device_is_registered(&sch->dev))
		device_unregister(&sch->dev);
189 190
	mutex_unlock(&sch->reg_mutex);
}
191
EXPORT_SYMBOL_GPL(css_sch_device_unregister);
192

193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237
static void ssd_from_pmcw(struct chsc_ssd_info *ssd, struct pmcw *pmcw)
{
	int i;
	int mask;

	memset(ssd, 0, sizeof(struct chsc_ssd_info));
	ssd->path_mask = pmcw->pim;
	for (i = 0; i < 8; i++) {
		mask = 0x80 >> i;
		if (pmcw->pim & mask) {
			chp_id_init(&ssd->chpid[i]);
			ssd->chpid[i].id = pmcw->chpid[i];
		}
	}
}

static void ssd_register_chpids(struct chsc_ssd_info *ssd)
{
	int i;
	int mask;

	for (i = 0; i < 8; i++) {
		mask = 0x80 >> i;
		if (ssd->path_mask & mask)
			if (!chp_is_registered(ssd->chpid[i]))
				chp_new(ssd->chpid[i]);
	}
}

void css_update_ssd_info(struct subchannel *sch)
{
	int ret;

	if (cio_is_console(sch->schid)) {
		/* Console is initialized too early for functions requiring
		 * memory allocation. */
		ssd_from_pmcw(&sch->ssd_info, &sch->schib.pmcw);
	} else {
		ret = chsc_get_ssd_info(sch->schid, &sch->ssd_info);
		if (ret)
			ssd_from_pmcw(&sch->ssd_info, &sch->schib.pmcw);
		ssd_register_chpids(&sch->ssd_info);
	}
}

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 267
static ssize_t type_show(struct device *dev, struct device_attribute *attr,
			 char *buf)
{
	struct subchannel *sch = to_subchannel(dev);

	return sprintf(buf, "%01x\n", sch->st);
}

static DEVICE_ATTR(type, 0444, type_show, NULL);

static ssize_t modalias_show(struct device *dev, struct device_attribute *attr,
			     char *buf)
{
	struct subchannel *sch = to_subchannel(dev);

	return sprintf(buf, "css:t%01X\n", sch->st);
}

static DEVICE_ATTR(modalias, 0444, modalias_show, NULL);

static struct attribute *subch_attrs[] = {
	&dev_attr_type.attr,
	&dev_attr_modalias.attr,
	NULL,
};

static struct attribute_group subch_attr_group = {
	.attrs = subch_attrs,
};

268
static const struct attribute_group *default_subch_attr_groups[] = {
269 270 271 272
	&subch_attr_group,
	NULL,
};

273
static int css_register_subchannel(struct subchannel *sch)
L
Linus Torvalds 已提交
274 275 276 277
{
	int ret;

	/* Initialize the subchannel structure */
278
	sch->dev.parent = &channel_subsystems[0]->device;
L
Linus Torvalds 已提交
279 280
	sch->dev.bus = &css_bus_type;
	sch->dev.release = &css_subchannel_release;
281
	sch->dev.groups = default_subch_attr_groups;
282 283 284 285 286
	/*
	 * We don't want to generate uevents for I/O subchannels that don't
	 * have a working ccw device behind them since they will be
	 * unregistered before they can be used anyway, so we delay the add
	 * uevent until after device recognition was successful.
287 288 289
	 * Note that we suppress the uevent for all subchannel types;
	 * the subchannel driver can decide itself when it wants to inform
	 * userspace of its existence.
290
	 */
291
	dev_set_uevent_suppress(&sch->dev, 1);
292
	css_update_ssd_info(sch);
L
Linus Torvalds 已提交
293
	/* make it known to the system */
294
	ret = css_sch_device_register(sch);
295
	if (ret) {
C
Cornelia Huck 已提交
296 297
		CIO_MSG_EVENT(0, "Could not register sch 0.%x.%04x: %d\n",
			      sch->schid.ssid, sch->schid.sch_no, ret);
298 299
		return ret;
	}
300 301 302 303 304 305
	if (!sch->driver) {
		/*
		 * No driver matched. Generate the uevent now so that
		 * a fitting driver module may be loaded based on the
		 * modalias.
		 */
306
		dev_set_uevent_suppress(&sch->dev, 0);
307 308
		kobject_uevent(&sch->dev.kobj, KOBJ_ADD);
	}
L
Linus Torvalds 已提交
309 310 311
	return ret;
}

C
Cornelia Huck 已提交
312
int css_probe_device(struct subchannel_id schid)
L
Linus Torvalds 已提交
313 314 315 316
{
	int ret;
	struct subchannel *sch;

317 318 319 320 321 322 323
	if (cio_is_console(schid))
		sch = cio_get_console_subchannel();
	else {
		sch = css_alloc_subchannel(schid);
		if (IS_ERR(sch))
			return PTR_ERR(sch);
	}
L
Linus Torvalds 已提交
324
	ret = css_register_subchannel(sch);
325 326 327 328
	if (ret) {
		if (!cio_is_console(schid))
			put_device(&sch->dev);
	}
L
Linus Torvalds 已提交
329 330 331
	return ret;
}

C
Cornelia Huck 已提交
332 333 334 335
static int
check_subchannel(struct device * dev, void * data)
{
	struct subchannel *sch;
336
	struct subchannel_id *schid = data;
C
Cornelia Huck 已提交
337 338

	sch = to_subchannel(dev);
339
	return schid_equal(&sch->schid, schid);
C
Cornelia Huck 已提交
340 341
}

L
Linus Torvalds 已提交
342
struct subchannel *
343
get_subchannel_by_schid(struct subchannel_id schid)
L
Linus Torvalds 已提交
344 345 346
{
	struct device *dev;

C
Cornelia Huck 已提交
347
	dev = bus_find_device(&css_bus_type, NULL,
348
			      &schid, check_subchannel);
L
Linus Torvalds 已提交
349

C
Cornelia Huck 已提交
350
	return dev ? to_subchannel(dev) : NULL;
L
Linus Torvalds 已提交
351 352
}

353 354 355 356 357 358 359 360
/**
 * css_sch_is_valid() - check if a subchannel is valid
 * @schib: subchannel information block for the subchannel
 */
int css_sch_is_valid(struct schib *schib)
{
	if ((schib->pmcw.st == SUBCHANNEL_TYPE_IO) && !schib->pmcw.dnv)
		return 0;
361 362
	if ((schib->pmcw.st == SUBCHANNEL_TYPE_MSG) && !schib->pmcw.w)
		return 0;
363 364 365 366
	return 1;
}
EXPORT_SYMBOL_GPL(css_sch_is_valid);

367 368 369 370 371 372 373 374
static int css_evaluate_new_subchannel(struct subchannel_id schid, int slow)
{
	struct schib schib;

	if (!slow) {
		/* Will be done on the slow path. */
		return -EAGAIN;
	}
375
	if (stsch_err(schid, &schib) || !css_sch_is_valid(&schib)) {
376 377
		/* Unusable - ignore. */
		return 0;
L
Linus Torvalds 已提交
378
	}
379 380 381 382 383 384
	CIO_MSG_EVENT(4, "Evaluating schid 0.%x.%04x, event %d, unknown, "
			 "slow path.\n", schid.ssid, schid.sch_no, CIO_OPER);

	return css_probe_device(schid);
}

C
Cornelia Huck 已提交
385 386 387 388 389 390 391 392 393 394 395 396 397 398 399
static int css_evaluate_known_subchannel(struct subchannel *sch, int slow)
{
	int ret = 0;

	if (sch->driver) {
		if (sch->driver->sch_event)
			ret = sch->driver->sch_event(sch, slow);
		else
			dev_dbg(&sch->dev,
				"Got subchannel machine check but "
				"no sch_event handler provided.\n");
	}
	return ret;
}

400
static void css_evaluate_subchannel(struct subchannel_id schid, int slow)
401 402 403 404 405 406 407 408 409 410
{
	struct subchannel *sch;
	int ret;

	sch = get_subchannel_by_schid(schid);
	if (sch) {
		ret = css_evaluate_known_subchannel(sch, slow);
		put_device(&sch->dev);
	} else
		ret = css_evaluate_new_subchannel(schid, slow);
411 412
	if (ret == -EAGAIN)
		css_schedule_eval(schid);
L
Linus Torvalds 已提交
413 414
}

415 416
static struct idset *slow_subchannel_set;
static spinlock_t slow_subchannel_lock;
417 418
static wait_queue_head_t css_eval_wq;
static atomic_t css_eval_scheduled;
419 420

static int __init slow_subchannel_init(void)
L
Linus Torvalds 已提交
421
{
422
	spin_lock_init(&slow_subchannel_lock);
423 424
	atomic_set(&css_eval_scheduled, 0);
	init_waitqueue_head(&css_eval_wq);
425 426
	slow_subchannel_set = idset_sch_new();
	if (!slow_subchannel_set) {
C
Cornelia Huck 已提交
427
		CIO_MSG_EVENT(0, "could not allocate slow subchannel set\n");
428 429 430
		return -ENOMEM;
	}
	return 0;
L
Linus Torvalds 已提交
431 432
}

433
static int slow_eval_known_fn(struct subchannel *sch, void *data)
L
Linus Torvalds 已提交
434
{
435 436
	int eval;
	int rc;
L
Linus Torvalds 已提交
437 438

	spin_lock_irq(&slow_subchannel_lock);
439 440 441 442 443 444 445
	eval = idset_sch_contains(slow_subchannel_set, sch->schid);
	idset_sch_del(slow_subchannel_set, sch->schid);
	spin_unlock_irq(&slow_subchannel_lock);
	if (eval) {
		rc = css_evaluate_known_subchannel(sch, 1);
		if (rc == -EAGAIN)
			css_schedule_eval(sch->schid);
L
Linus Torvalds 已提交
446
	}
447 448 449 450 451 452 453 454 455 456 457
	return 0;
}

static int slow_eval_unknown_fn(struct subchannel_id schid, void *data)
{
	int eval;
	int rc = 0;

	spin_lock_irq(&slow_subchannel_lock);
	eval = idset_sch_contains(slow_subchannel_set, schid);
	idset_sch_del(slow_subchannel_set, schid);
L
Linus Torvalds 已提交
458
	spin_unlock_irq(&slow_subchannel_lock);
459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479
	if (eval) {
		rc = css_evaluate_new_subchannel(schid, 1);
		switch (rc) {
		case -EAGAIN:
			css_schedule_eval(schid);
			rc = 0;
			break;
		case -ENXIO:
		case -ENOMEM:
		case -EIO:
			/* These should abort looping */
			break;
		default:
			rc = 0;
		}
	}
	return rc;
}

static void css_slow_path_func(struct work_struct *unused)
{
480 481
	unsigned long flags;

482 483 484
	CIO_TRACE_EVENT(4, "slowpath");
	for_each_subchannel_staged(slow_eval_known_fn, slow_eval_unknown_fn,
				   NULL);
485 486 487 488 489 490
	spin_lock_irqsave(&slow_subchannel_lock, flags);
	if (idset_is_empty(slow_subchannel_set)) {
		atomic_set(&css_eval_scheduled, 0);
		wake_up(&css_eval_wq);
	}
	spin_unlock_irqrestore(&slow_subchannel_lock, flags);
L
Linus Torvalds 已提交
491 492
}

493
static DECLARE_WORK(slow_path_work, css_slow_path_func);
L
Linus Torvalds 已提交
494 495
struct workqueue_struct *slow_path_wq;

496 497 498 499 500 501
void css_schedule_eval(struct subchannel_id schid)
{
	unsigned long flags;

	spin_lock_irqsave(&slow_subchannel_lock, flags);
	idset_sch_add(slow_subchannel_set, schid);
502
	atomic_set(&css_eval_scheduled, 1);
503 504 505 506 507 508 509 510 511 512
	queue_work(slow_path_wq, &slow_path_work);
	spin_unlock_irqrestore(&slow_subchannel_lock, flags);
}

void css_schedule_eval_all(void)
{
	unsigned long flags;

	spin_lock_irqsave(&slow_subchannel_lock, flags);
	idset_fill(slow_subchannel_set);
513
	atomic_set(&css_eval_scheduled, 1);
514 515 516 517
	queue_work(slow_path_wq, &slow_path_work);
	spin_unlock_irqrestore(&slow_subchannel_lock, flags);
}

518
static int __unset_registered(struct device *dev, void *data)
519
{
520 521
	struct idset *set = data;
	struct subchannel *sch = to_subchannel(dev);
522

523 524
	idset_sch_del(set, sch->schid);
	return 0;
525 526
}

527
void css_schedule_eval_all_unreg(void)
528
{
529 530
	unsigned long flags;
	struct idset *unreg_set;
531

532 533 534 535 536
	/* Find unregistered subchannels. */
	unreg_set = idset_sch_new();
	if (!unreg_set) {
		/* Fallback. */
		css_schedule_eval_all();
537 538
		return;
	}
539 540 541 542 543 544 545 546 547
	idset_fill(unreg_set);
	bus_for_each_dev(&css_bus_type, NULL, unreg_set, __unset_registered);
	/* Apply to slow_subchannel_set. */
	spin_lock_irqsave(&slow_subchannel_lock, flags);
	idset_add_set(slow_subchannel_set, unreg_set);
	atomic_set(&css_eval_scheduled, 1);
	queue_work(slow_path_wq, &slow_path_work);
	spin_unlock_irqrestore(&slow_subchannel_lock, flags);
	idset_free(unreg_set);
548 549
}

550 551 552 553
void css_wait_for_slow_path(void)
{
	flush_workqueue(slow_path_wq);
}
554 555 556 557

/* Schedule reprobing of all unregistered subchannels. */
void css_schedule_reprobe(void)
{
558
	css_schedule_eval_all_unreg();
559 560 561
}
EXPORT_SYMBOL_GPL(css_schedule_reprobe);

L
Linus Torvalds 已提交
562 563 564
/*
 * Called from the machine check handler for subchannel report words.
 */
565
static void css_process_crw(struct crw *crw0, struct crw *crw1, int overflow)
L
Linus Torvalds 已提交
566
{
567
	struct subchannel_id mchk_schid;
L
Linus Torvalds 已提交
568

569 570 571 572 573 574 575 576 577 578 579 580 581
	if (overflow) {
		css_schedule_eval_all();
		return;
	}
	CIO_CRW_EVENT(2, "CRW0 reports slct=%d, oflw=%d, "
		      "chn=%d, rsc=%X, anc=%d, erc=%X, rsid=%X\n",
		      crw0->slct, crw0->oflw, crw0->chn, crw0->rsc, crw0->anc,
		      crw0->erc, crw0->rsid);
	if (crw1)
		CIO_CRW_EVENT(2, "CRW1 reports slct=%d, oflw=%d, "
			      "chn=%d, rsc=%X, anc=%d, erc=%X, rsid=%X\n",
			      crw1->slct, crw1->oflw, crw1->chn, crw1->rsc,
			      crw1->anc, crw1->erc, crw1->rsid);
582
	init_subchannel_id(&mchk_schid);
583 584 585
	mchk_schid.sch_no = crw0->rsid;
	if (crw1)
		mchk_schid.ssid = (crw1->rsid >> 8) & 3;
586

587
	/*
L
Linus Torvalds 已提交
588 589 590 591
	 * Since we are always presented with IPI in the CRW, we have to
	 * use stsch() to find out if the subchannel in question has come
	 * or gone.
	 */
592
	css_evaluate_subchannel(mchk_schid, 0);
L
Linus Torvalds 已提交
593 594 595
}

static void __init
596
css_generate_pgid(struct channel_subsystem *css, u32 tod_high)
L
Linus Torvalds 已提交
597
{
598
	if (css_general_characteristics.mcss) {
599 600 601
		css->global_pgid.pgid_high.ext_cssid.version = 0x80;
		css->global_pgid.pgid_high.ext_cssid.cssid = css->cssid;
	} else {
L
Linus Torvalds 已提交
602
#ifdef CONFIG_SMP
603
		css->global_pgid.pgid_high.cpu_addr = stap();
L
Linus Torvalds 已提交
604
#else
605
		css->global_pgid.pgid_high.cpu_addr = 0;
L
Linus Torvalds 已提交
606 607
#endif
	}
608 609
	css->global_pgid.cpu_id = S390_lowcore.cpu_id.ident;
	css->global_pgid.cpu_model = S390_lowcore.cpu_id.machine;
610 611 612 613
	css->global_pgid.tod_high = tod_high;

}

614 615 616 617 618 619
static void
channel_subsystem_release(struct device *dev)
{
	struct channel_subsystem *css;

	css = to_css(dev);
620
	mutex_destroy(&css->mutex);
621 622 623 624 625
	if (css->pseudo_subchannel) {
		/* Implies that it has been generated but never registered. */
		css_subchannel_release(&css->pseudo_subchannel->dev);
		css->pseudo_subchannel = NULL;
	}
626 627 628
	kfree(css);
}

629 630 631 632 633
static ssize_t
css_cm_enable_show(struct device *dev, struct device_attribute *attr,
		   char *buf)
{
	struct channel_subsystem *css = to_css(dev);
634
	int ret;
635 636 637

	if (!css)
		return 0;
638 639 640 641
	mutex_lock(&css->mutex);
	ret = sprintf(buf, "%x\n", css->cm_enabled);
	mutex_unlock(&css->mutex);
	return ret;
642 643 644 645 646 647 648 649
}

static ssize_t
css_cm_enable_store(struct device *dev, struct device_attribute *attr,
		    const char *buf, size_t count)
{
	struct channel_subsystem *css = to_css(dev);
	int ret;
650
	unsigned long val;
651

652 653 654
	ret = strict_strtoul(buf, 16, &val);
	if (ret)
		return ret;
655
	mutex_lock(&css->mutex);
656 657
	switch (val) {
	case 0:
658 659
		ret = css->cm_enabled ? chsc_secm(css, 0) : 0;
		break;
660
	case 1:
661 662 663 664 665
		ret = css->cm_enabled ? 0 : chsc_secm(css, 1);
		break;
	default:
		ret = -EINVAL;
	}
666
	mutex_unlock(&css->mutex);
667 668 669 670 671
	return ret < 0 ? ret : count;
}

static DEVICE_ATTR(cm_enable, 0644, css_cm_enable_show, css_cm_enable_store);

672
static int __init setup_css(int nr)
673 674
{
	u32 tod_high;
675
	int ret;
676
	struct channel_subsystem *css;
677

678 679 680 681 682
	css = channel_subsystems[nr];
	memset(css, 0, sizeof(struct channel_subsystem));
	css->pseudo_subchannel =
		kzalloc(sizeof(*css->pseudo_subchannel), GFP_KERNEL);
	if (!css->pseudo_subchannel)
683
		return -ENOMEM;
684 685
	css->pseudo_subchannel->dev.parent = &css->device;
	css->pseudo_subchannel->dev.release = css_subchannel_release;
686
	dev_set_name(&css->pseudo_subchannel->dev, "defunct");
687
	ret = cio_create_sch_lock(css->pseudo_subchannel);
688
	if (ret) {
689
		kfree(css->pseudo_subchannel);
690 691
		return ret;
	}
692 693 694
	mutex_init(&css->mutex);
	css->valid = 1;
	css->cssid = nr;
695
	dev_set_name(&css->device, "css%x", nr);
696
	css->device.release = channel_subsystem_release;
697
	tod_high = (u32) (get_clock() >> 32);
698
	css_generate_pgid(css, tod_high);
699
	return 0;
L
Linus Torvalds 已提交
700 701
}

702 703 704 705 706 707 708 709 710 711 712
static int css_reboot_event(struct notifier_block *this,
			    unsigned long event,
			    void *ptr)
{
	int ret, i;

	ret = NOTIFY_DONE;
	for (i = 0; i <= __MAX_CSSID; i++) {
		struct channel_subsystem *css;

		css = channel_subsystems[i];
713
		mutex_lock(&css->mutex);
714 715 716
		if (css->cm_enabled)
			if (chsc_secm(css, 0))
				ret = NOTIFY_BAD;
717
		mutex_unlock(&css->mutex);
718 719 720 721 722 723 724 725 726
	}

	return ret;
}

static struct notifier_block css_reboot_notifier = {
	.notifier_call = css_reboot_event,
};

727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799
/*
 * Since the css devices are neither on a bus nor have a class
 * nor have a special device type, we cannot stop/restart channel
 * path measurements via the normal suspend/resume callbacks, but have
 * to use notifiers.
 */
static int css_power_event(struct notifier_block *this, unsigned long event,
			   void *ptr)
{
	void *secm_area;
	int ret, i;

	switch (event) {
	case PM_HIBERNATION_PREPARE:
	case PM_SUSPEND_PREPARE:
		ret = NOTIFY_DONE;
		for (i = 0; i <= __MAX_CSSID; i++) {
			struct channel_subsystem *css;

			css = channel_subsystems[i];
			mutex_lock(&css->mutex);
			if (!css->cm_enabled) {
				mutex_unlock(&css->mutex);
				continue;
			}
			secm_area = (void *)get_zeroed_page(GFP_KERNEL |
							    GFP_DMA);
			if (secm_area) {
				if (__chsc_do_secm(css, 0, secm_area))
					ret = NOTIFY_BAD;
				free_page((unsigned long)secm_area);
			} else
				ret = NOTIFY_BAD;

			mutex_unlock(&css->mutex);
		}
		break;
	case PM_POST_HIBERNATION:
	case PM_POST_SUSPEND:
		ret = NOTIFY_DONE;
		for (i = 0; i <= __MAX_CSSID; i++) {
			struct channel_subsystem *css;

			css = channel_subsystems[i];
			mutex_lock(&css->mutex);
			if (!css->cm_enabled) {
				mutex_unlock(&css->mutex);
				continue;
			}
			secm_area = (void *)get_zeroed_page(GFP_KERNEL |
							    GFP_DMA);
			if (secm_area) {
				if (__chsc_do_secm(css, 1, secm_area))
					ret = NOTIFY_BAD;
				free_page((unsigned long)secm_area);
			} else
				ret = NOTIFY_BAD;

			mutex_unlock(&css->mutex);
		}
		/* search for subchannels, which appeared during hibernation */
		css_schedule_reprobe();
		break;
	default:
		ret = NOTIFY_DONE;
	}
	return ret;

}
static struct notifier_block css_power_notifier = {
	.notifier_call = css_power_event,
};

L
Linus Torvalds 已提交
800 801 802 803 804
/*
 * Now that the driver core is running, we can setup our channel subsystem.
 * The struct subchannel's are created during probing (except for the
 * static console subchannel).
 */
S
Sebastian Ott 已提交
805
static int __init css_bus_init(void)
L
Linus Torvalds 已提交
806
{
807
	int ret, i;
L
Linus Torvalds 已提交
808

809 810
	ret = chsc_determine_css_characteristics();
	if (ret == -ENOMEM)
S
Sebastian Ott 已提交
811
		goto out;
L
Linus Torvalds 已提交
812

813 814 815 816
	ret = chsc_alloc_sei_area();
	if (ret)
		goto out;

817 818 819 820 821 822 823 824 825 826 827 828
	/* Try to enable MSS. */
	ret = chsc_enable_facility(CHSC_SDA_OC_MSS);
	switch (ret) {
	case 0: /* Success. */
		max_ssid = __MAX_SSID;
		break;
	case -ENOMEM:
		goto out;
	default:
		max_ssid = 0;
	}

829 830 831 832
	ret = slow_subchannel_init();
	if (ret)
		goto out;

833
	ret = crw_register_handler(CRW_RSC_SCH, css_process_crw);
834 835 836
	if (ret)
		goto out;

L
Linus Torvalds 已提交
837 838 839
	if ((ret = bus_register(&css_bus_type)))
		goto out;

840 841
	/* Setup css structure. */
	for (i = 0; i <= __MAX_CSSID; i++) {
842 843 844 845
		struct channel_subsystem *css;

		css = kmalloc(sizeof(struct channel_subsystem), GFP_KERNEL);
		if (!css) {
846
			ret = -ENOMEM;
847
			goto out_unregister;
848
		}
849
		channel_subsystems[i] = css;
850
		ret = setup_css(i);
851 852 853 854
		if (ret) {
			kfree(channel_subsystems[i]);
			goto out_unregister;
		}
855
		ret = device_register(&css->device);
856 857 858 859
		if (ret) {
			put_device(&css->device);
			goto out_unregister;
		}
860
		if (css_chsc_characteristics.secm) {
861
			ret = device_create_file(&css->device,
862 863 864 865
						 &dev_attr_cm_enable);
			if (ret)
				goto out_device;
		}
866
		ret = device_register(&css->pseudo_subchannel->dev);
867 868
		if (ret) {
			put_device(&css->pseudo_subchannel->dev);
869
			goto out_file;
870
		}
871
	}
872 873
	ret = register_reboot_notifier(&css_reboot_notifier);
	if (ret)
874
		goto out_unregister;
875 876 877 878 879
	ret = register_pm_notifier(&css_power_notifier);
	if (ret) {
		unregister_reboot_notifier(&css_reboot_notifier);
		goto out_unregister;
	}
L
Linus Torvalds 已提交
880 881
	css_init_done = 1;

882
	/* Enable default isc for I/O subchannels. */
883
	isc_register(IO_SCH_ISC);
L
Linus Torvalds 已提交
884 885

	return 0;
886
out_file:
887 888 889
	if (css_chsc_characteristics.secm)
		device_remove_file(&channel_subsystems[i]->device,
				   &dev_attr_cm_enable);
890
out_device:
891
	device_unregister(&channel_subsystems[i]->device);
892
out_unregister:
893
	while (i > 0) {
894 895
		struct channel_subsystem *css;

896
		i--;
897 898
		css = channel_subsystems[i];
		device_unregister(&css->pseudo_subchannel->dev);
899
		css->pseudo_subchannel = NULL;
900
		if (css_chsc_characteristics.secm)
901
			device_remove_file(&css->device,
902
					   &dev_attr_cm_enable);
903
		device_unregister(&css->device);
904
	}
L
Linus Torvalds 已提交
905 906
	bus_unregister(&css_bus_type);
out:
907
	crw_unregister_handler(CRW_RSC_CSS);
908
	chsc_free_sei_area();
909
	idset_free(slow_subchannel_set);
910 911
	pr_alert("The CSS device driver initialization failed with "
		 "errno=%d\n", ret);
L
Linus Torvalds 已提交
912 913 914
	return ret;
}

S
Sebastian Ott 已提交
915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930
static void __init css_bus_cleanup(void)
{
	struct channel_subsystem *css;
	int i;

	for (i = 0; i <= __MAX_CSSID; i++) {
		css = channel_subsystems[i];
		device_unregister(&css->pseudo_subchannel->dev);
		css->pseudo_subchannel = NULL;
		if (css_chsc_characteristics.secm)
			device_remove_file(&css->device, &dev_attr_cm_enable);
		device_unregister(&css->device);
	}
	bus_unregister(&css_bus_type);
	crw_unregister_handler(CRW_RSC_CSS);
	chsc_free_sei_area();
931
	idset_free(slow_subchannel_set);
S
Sebastian Ott 已提交
932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950
	isc_unregister(IO_SCH_ISC);
}

static int __init channel_subsystem_init(void)
{
	int ret;

	ret = css_bus_init();
	if (ret)
		return ret;

	ret = io_subchannel_init();
	if (ret)
		css_bus_cleanup();

	return ret;
}
subsys_initcall(channel_subsystem_init);

S
Sebastian Ott 已提交
951 952 953 954 955 956 957 958 959
static int css_settle(struct device_driver *drv, void *unused)
{
	struct css_driver *cssdrv = to_cssdriver(drv);

	if (cssdrv->settle)
		cssdrv->settle();
	return 0;
}

S
Sebastian Ott 已提交
960 961 962 963 964 965
/*
 * Wait for the initialization of devices to finish, to make sure we are
 * done with our setup if the search for the root device starts.
 */
static int __init channel_subsystem_init_sync(void)
{
966 967
	/* Start initial subchannel evaluation. */
	css_schedule_eval_all();
968 969
	/* Wait for the evaluation of subchannels to finish. */
	wait_event(css_eval_wq, atomic_read(&css_eval_scheduled) == 0);
S
Sebastian Ott 已提交
970 971
	/* Wait for the subchannel type specific initialization to finish */
	return bus_for_each_drv(&css_bus_type, NULL, NULL, css_settle);
S
Sebastian Ott 已提交
972 973 974
}
subsys_initcall_sync(channel_subsystem_init_sync);

975 976 977 978 979
int sch_is_pseudo_sch(struct subchannel *sch)
{
	return sch == to_css(sch->dev.parent)->pseudo_subchannel;
}

980
static int css_bus_match(struct device *dev, struct device_driver *drv)
L
Linus Torvalds 已提交
981
{
982 983
	struct subchannel *sch = to_subchannel(dev);
	struct css_driver *driver = to_cssdriver(drv);
984
	struct css_device_id *id;
L
Linus Torvalds 已提交
985

986 987 988 989
	for (id = driver->subchannel_type; id->match_flags; id++) {
		if (sch->st == id->type)
			return 1;
	}
L
Linus Torvalds 已提交
990 991 992 993

	return 0;
}

C
Cornelia Huck 已提交
994
static int css_probe(struct device *dev)
995 996
{
	struct subchannel *sch;
C
Cornelia Huck 已提交
997
	int ret;
998 999

	sch = to_subchannel(dev);
1000
	sch->driver = to_cssdriver(dev->driver);
C
Cornelia Huck 已提交
1001 1002 1003 1004
	ret = sch->driver->probe ? sch->driver->probe(sch) : 0;
	if (ret)
		sch->driver = NULL;
	return ret;
1005 1006
}

C
Cornelia Huck 已提交
1007
static int css_remove(struct device *dev)
1008 1009
{
	struct subchannel *sch;
C
Cornelia Huck 已提交
1010
	int ret;
1011 1012

	sch = to_subchannel(dev);
C
Cornelia Huck 已提交
1013 1014 1015
	ret = sch->driver->remove ? sch->driver->remove(sch) : 0;
	sch->driver = NULL;
	return ret;
1016 1017
}

C
Cornelia Huck 已提交
1018
static void css_shutdown(struct device *dev)
1019 1020 1021 1022
{
	struct subchannel *sch;

	sch = to_subchannel(dev);
C
Cornelia Huck 已提交
1023
	if (sch->driver && sch->driver->shutdown)
1024 1025 1026
		sch->driver->shutdown(sch);
}

1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037 1038
static int css_uevent(struct device *dev, struct kobj_uevent_env *env)
{
	struct subchannel *sch = to_subchannel(dev);
	int ret;

	ret = add_uevent_var(env, "ST=%01X", sch->st);
	if (ret)
		return ret;
	ret = add_uevent_var(env, "MODALIAS=css:t%01X", sch->st);
	return ret;
}

1039 1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105
static int css_pm_prepare(struct device *dev)
{
	struct subchannel *sch = to_subchannel(dev);
	struct css_driver *drv;

	if (mutex_is_locked(&sch->reg_mutex))
		return -EAGAIN;
	if (!sch->dev.driver)
		return 0;
	drv = to_cssdriver(sch->dev.driver);
	/* Notify drivers that they may not register children. */
	return drv->prepare ? drv->prepare(sch) : 0;
}

static void css_pm_complete(struct device *dev)
{
	struct subchannel *sch = to_subchannel(dev);
	struct css_driver *drv;

	if (!sch->dev.driver)
		return;
	drv = to_cssdriver(sch->dev.driver);
	if (drv->complete)
		drv->complete(sch);
}

static int css_pm_freeze(struct device *dev)
{
	struct subchannel *sch = to_subchannel(dev);
	struct css_driver *drv;

	if (!sch->dev.driver)
		return 0;
	drv = to_cssdriver(sch->dev.driver);
	return drv->freeze ? drv->freeze(sch) : 0;
}

static int css_pm_thaw(struct device *dev)
{
	struct subchannel *sch = to_subchannel(dev);
	struct css_driver *drv;

	if (!sch->dev.driver)
		return 0;
	drv = to_cssdriver(sch->dev.driver);
	return drv->thaw ? drv->thaw(sch) : 0;
}

static int css_pm_restore(struct device *dev)
{
	struct subchannel *sch = to_subchannel(dev);
	struct css_driver *drv;

	if (!sch->dev.driver)
		return 0;
	drv = to_cssdriver(sch->dev.driver);
	return drv->restore ? drv->restore(sch) : 0;
}

static struct dev_pm_ops css_pm_ops = {
	.prepare = css_pm_prepare,
	.complete = css_pm_complete,
	.freeze = css_pm_freeze,
	.thaw = css_pm_thaw,
	.restore = css_pm_restore,
};

L
Linus Torvalds 已提交
1106
struct bus_type css_bus_type = {
1107 1108 1109 1110 1111
	.name     = "css",
	.match    = css_bus_match,
	.probe    = css_probe,
	.remove   = css_remove,
	.shutdown = css_shutdown,
1112
	.uevent   = css_uevent,
1113
	.pm = &css_pm_ops,
L
Linus Torvalds 已提交
1114 1115
};

1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126
/**
 * css_driver_register - register a css driver
 * @cdrv: css driver to register
 *
 * This is mainly a wrapper around driver_register that sets name
 * and bus_type in the embedded struct device_driver correctly.
 */
int css_driver_register(struct css_driver *cdrv)
{
	cdrv->drv.name = cdrv->name;
	cdrv->drv.bus = &css_bus_type;
1127
	cdrv->drv.owner = cdrv->owner;
1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143
	return driver_register(&cdrv->drv);
}
EXPORT_SYMBOL_GPL(css_driver_register);

/**
 * css_driver_unregister - unregister a css driver
 * @cdrv: css driver to unregister
 *
 * This is a wrapper around driver_unregister.
 */
void css_driver_unregister(struct css_driver *cdrv)
{
	driver_unregister(&cdrv->drv);
}
EXPORT_SYMBOL_GPL(css_driver_unregister);

L
Linus Torvalds 已提交
1144 1145
MODULE_LICENSE("GPL");
EXPORT_SYMBOL(css_bus_type);