ide-proc.c 17.5 KB
Newer Older
L
Linus Torvalds 已提交
1 2
/*
 *  Copyright (C) 1997-1998	Mark Lord
A
Alan Cox 已提交
3
 *  Copyright (C) 2003		Red Hat
4 5
 *
 *  Some code was moved here from ide.c, see it for original copyrights.
L
Linus Torvalds 已提交
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
 */

/*
 * This is the /proc/ide/ filesystem implementation.
 *
 * Drive/Driver settings can be retrieved by reading the drive's
 * "settings" files.  e.g.    "cat /proc/ide0/hda/settings"
 * To write a new value "val" into a specific setting "name", use:
 *   echo "name:val" >/proc/ide/ide0/hda/settings
 */

#include <linux/module.h>

#include <asm/uaccess.h>
#include <linux/errno.h>
#include <linux/proc_fs.h>
#include <linux/stat.h>
#include <linux/mm.h>
#include <linux/pci.h>
#include <linux/ctype.h>
#include <linux/ide.h>
#include <linux/seq_file.h>

#include <asm/io.h>

31 32
static struct proc_dir_entry *proc_ide_root;

L
Linus Torvalds 已提交
33 34 35 36 37 38 39 40
static int proc_ide_read_imodel
	(char *page, char **start, off_t off, int count, int *eof, void *data)
{
	ide_hwif_t	*hwif = (ide_hwif_t *) data;
	int		len;
	const char	*name;

	switch (hwif->chipset) {
41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
	case ide_generic:	name = "generic";	break;
	case ide_pci:		name = "pci";		break;
	case ide_cmd640:	name = "cmd640";	break;
	case ide_dtc2278:	name = "dtc2278";	break;
	case ide_ali14xx:	name = "ali14xx";	break;
	case ide_qd65xx:	name = "qd65xx";	break;
	case ide_umc8672:	name = "umc8672";	break;
	case ide_ht6560b:	name = "ht6560b";	break;
	case ide_rz1000:	name = "rz1000";	break;
	case ide_trm290:	name = "trm290";	break;
	case ide_cy82c693:	name = "cy82c693";	break;
	case ide_4drives:	name = "4drives";	break;
	case ide_pmac:		name = "mac-io";	break;
	case ide_au1xxx:	name = "au1xxx";	break;
	case ide_palm3710:      name = "palm3710";      break;
	case ide_acorn:		name = "acorn";		break;
	default:		name = "(unknown)";	break;
L
Linus Torvalds 已提交
58 59
	}
	len = sprintf(page, "%s\n", name);
60
	PROC_IDE_READ_RETURN(page, start, off, count, eof, len);
L
Linus Torvalds 已提交
61 62 63 64 65 66 67 68
}

static int proc_ide_read_mate
	(char *page, char **start, off_t off, int count, int *eof, void *data)
{
	ide_hwif_t	*hwif = (ide_hwif_t *) data;
	int		len;

69
	if (hwif && hwif->mate)
L
Linus Torvalds 已提交
70 71 72
		len = sprintf(page, "%s\n", hwif->mate->name);
	else
		len = sprintf(page, "(none)\n");
73
	PROC_IDE_READ_RETURN(page, start, off, count, eof, len);
L
Linus Torvalds 已提交
74 75 76 77 78 79 80 81 82 83 84
}

static int proc_ide_read_channel
	(char *page, char **start, off_t off, int count, int *eof, void *data)
{
	ide_hwif_t	*hwif = (ide_hwif_t *) data;
	int		len;

	page[0] = hwif->channel ? '1' : '0';
	page[1] = '\n';
	len = 2;
85
	PROC_IDE_READ_RETURN(page, start, off, count, eof, len);
L
Linus Torvalds 已提交
86 87 88 89 90 91 92 93 94 95 96 97
}

static int proc_ide_read_identify
	(char *page, char **start, off_t off, int count, int *eof, void *data)
{
	ide_drive_t	*drive = (ide_drive_t *)data;
	int		len = 0, i = 0;
	int		err = 0;

	len = sprintf(page, "\n");

	if (drive) {
98
		__le16 *val = (__le16 *)page;
L
Linus Torvalds 已提交
99 100 101

		err = taskfile_lib_get_identify(drive, page);
		if (!err) {
102 103
			char *out = (char *)page + SECTOR_SIZE;

L
Linus Torvalds 已提交
104 105 106
			page = out;
			do {
				out += sprintf(out, "%04x%c",
107
					le16_to_cpup(val), (++i & 7) ? ' ' : '\n');
L
Linus Torvalds 已提交
108
				val += 1;
109
			} while (i < SECTOR_SIZE / 2);
L
Linus Torvalds 已提交
110 111 112
			len = out - page;
		}
	}
113
	PROC_IDE_READ_RETURN(page, start, off, count, eof, len);
L
Linus Torvalds 已提交
114 115
}

116
/**
117 118
 *	ide_find_setting	-	find a specific setting
 *	@st: setting table pointer
119 120
 *	@name: setting name
 *
121
 *	Scan's the setting table for a matching entry and returns
122 123 124 125
 *	this or NULL if no entry is found. The caller must hold the
 *	setting semaphore
 */

126 127 128
static
const struct ide_proc_devset *ide_find_setting(const struct ide_proc_devset *st,
					       char *name)
129
{
130 131
	while (st->name) {
		if (strcmp(st->name, name) == 0)
132
			break;
133
		st++;
134
	}
135
	return st->name ? st : NULL;
136 137 138 139 140 141 142 143
}

/**
 *	ide_read_setting	-	read an IDE setting
 *	@drive: drive to read from
 *	@setting: drive setting
 *
 *	Read a drive setting and return the value. The caller
144
 *	must hold the ide_setting_mtx when making this call.
145 146 147 148 149 150
 *
 *	BUGS: the data return and error are the same return value
 *	so an error -EINVAL and true return of the same value cannot
 *	be told apart
 */

151
static int ide_read_setting(ide_drive_t *drive,
152
			    const struct ide_proc_devset *setting)
153
{
154
	const struct ide_devset *ds = setting->setting;
155 156
	int val = -EINVAL;

157
	if (ds->get)
158
		val = ds->get(drive);
159

160 161 162 163 164 165 166 167 168 169
	return val;
}

/**
 *	ide_write_setting	-	read an IDE setting
 *	@drive: drive to read from
 *	@setting: drive setting
 *	@val: value
 *
 *	Write a drive setting if it is possible. The caller
170
 *	must hold the ide_setting_mtx when making this call.
171 172 173 174 175 176 177 178 179 180
 *
 *	BUGS: the data return and error are the same return value
 *	so an error -EINVAL and true return of the same value cannot
 *	be told apart
 *
 *	FIXME:  This should be changed to enqueue a special request
 *	to the driver to change settings, and then wait on a sema for completion.
 *	The current scheme of polling is kludgy, though safe enough.
 */

181
static int ide_write_setting(ide_drive_t *drive,
182
			     const struct ide_proc_devset *setting, int val)
183
{
184 185
	const struct ide_devset *ds = setting->setting;

186 187
	if (!capable(CAP_SYS_ADMIN))
		return -EACCES;
188
	if (!ds->set)
189
		return -EPERM;
190 191
	if ((ds->flags & DS_SYNC)
	    && (val < setting->min || val > setting->max))
192
		return -EINVAL;
193
	return ide_devset_execute(drive, ds, val);
194 195
}

196
ide_devset_get(xfer_rate, current_speed);
197

198 199
static int set_xfer_rate (ide_drive_t *drive, int arg)
{
200
	ide_task_t task;
201 202
	int err;

203
	if (arg < XFER_PIO_0 || arg > XFER_UDMA_6)
204 205
		return -EINVAL;

206
	memset(&task, 0, sizeof(task));
207
	task.tf.command = ATA_CMD_SET_FEATURES;
208 209 210 211 212 213
	task.tf.feature = SETFEATURES_XFER;
	task.tf.nsect   = (u8)arg;
	task.tf_flags = IDE_TFLAG_OUT_FEATURE | IDE_TFLAG_OUT_NSECT |
			IDE_TFLAG_IN_NSECT;

	err = ide_no_data_taskfile(drive, &task);
214

215
	if (!err) {
216 217 218 219 220 221
		ide_set_xfer_rate(drive, (u8) arg);
		ide_driveid_update(drive);
	}
	return err;
}

222 223
ide_devset_rw(current_speed, xfer_rate);
ide_devset_rw_field(init_speed, init_speed);
B
Bartlomiej Zolnierkiewicz 已提交
224
ide_devset_rw_flag(nice1, IDE_DFLAG_NICE1);
225 226 227 228 229 230 231 232 233 234 235 236 237
ide_devset_rw_field(number, dn);

static const struct ide_proc_devset ide_generic_settings[] = {
	IDE_PROC_DEVSET(current_speed, 0, 70),
	IDE_PROC_DEVSET(init_speed, 0, 70),
	IDE_PROC_DEVSET(io_32bit,  0, 1 + (SUPPORT_VLB_SYNC << 1)),
	IDE_PROC_DEVSET(keepsettings, 0, 1),
	IDE_PROC_DEVSET(nice1, 0, 1),
	IDE_PROC_DEVSET(number, 0, 3),
	IDE_PROC_DEVSET(pio_mode, 0, 255),
	IDE_PROC_DEVSET(unmaskirq, 0, 1),
	IDE_PROC_DEVSET(using_dma, 0, 1),
	{ 0 },
238
};
239

L
Linus Torvalds 已提交
240 241
static void proc_ide_settings_warn(void)
{
242
	static int warned;
L
Linus Torvalds 已提交
243 244 245 246 247 248 249 250 251 252 253 254

	if (warned)
		return;

	printk(KERN_WARNING "Warning: /proc/ide/hd?/settings interface is "
			    "obsolete, and will be removed soon!\n");
	warned = 1;
}

static int proc_ide_read_settings
	(char *page, char **start, off_t off, int count, int *eof, void *data)
{
255 256
	const struct ide_proc_devset *setting, *g, *d;
	const struct ide_devset *ds;
L
Linus Torvalds 已提交
257 258 259 260 261 262
	ide_drive_t	*drive = (ide_drive_t *) data;
	char		*out = page;
	int		len, rc, mul_factor, div_factor;

	proc_ide_settings_warn();

263
	mutex_lock(&ide_setting_mtx);
264 265
	g = ide_generic_settings;
	d = drive->settings;
L
Linus Torvalds 已提交
266 267
	out += sprintf(out, "name\t\t\tvalue\t\tmin\t\tmax\t\tmode\n");
	out += sprintf(out, "----\t\t\t-----\t\t---\t\t---\t\t----\n");
268
	while (g->name || (d && d->name)) {
269
		/* read settings in the alphabetical order */
270 271 272
		if (g->name && d && d->name) {
			if (strcmp(d->name, g->name) < 0)
				setting = d++;
273
			else
274 275 276
				setting = g++;
		} else if (d && d->name) {
			setting = d++;
277
		} else
278
			setting = g++;
279 280
		mul_factor = setting->mulf ? setting->mulf(drive) : 1;
		div_factor = setting->divf ? setting->divf(drive) : 1;
L
Linus Torvalds 已提交
281
		out += sprintf(out, "%-24s", setting->name);
282 283
		rc = ide_read_setting(drive, setting);
		if (rc >= 0)
L
Linus Torvalds 已提交
284 285 286 287
			out += sprintf(out, "%-16d", rc * mul_factor / div_factor);
		else
			out += sprintf(out, "%-16s", "write-only");
		out += sprintf(out, "%-16d%-16d", (setting->min * mul_factor + div_factor - 1) / div_factor, setting->max * mul_factor / div_factor);
288 289
		ds = setting->setting;
		if (ds->get)
L
Linus Torvalds 已提交
290
			out += sprintf(out, "r");
291
		if (ds->set)
L
Linus Torvalds 已提交
292 293 294 295
			out += sprintf(out, "w");
		out += sprintf(out, "\n");
	}
	len = out - page;
296
	mutex_unlock(&ide_setting_mtx);
297
	PROC_IDE_READ_RETURN(page, start, off, count, eof, len);
L
Linus Torvalds 已提交
298 299 300 301 302 303 304 305 306
}

#define MAX_LEN	30

static int proc_ide_write_settings(struct file *file, const char __user *buffer,
				   unsigned long count, void *data)
{
	ide_drive_t	*drive = (ide_drive_t *) data;
	char		name[MAX_LEN + 1];
307
	int		for_real = 0, mul_factor, div_factor;
L
Linus Torvalds 已提交
308
	unsigned long	n;
309

310
	const struct ide_proc_devset *setting;
L
Linus Torvalds 已提交
311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376
	char *buf, *s;

	if (!capable(CAP_SYS_ADMIN))
		return -EACCES;

	proc_ide_settings_warn();

	if (count >= PAGE_SIZE)
		return -EINVAL;

	s = buf = (char *)__get_free_page(GFP_USER);
	if (!buf)
		return -ENOMEM;

	if (copy_from_user(buf, buffer, count)) {
		free_page((unsigned long)buf);
		return -EFAULT;
	}

	buf[count] = '\0';

	/*
	 * Skip over leading whitespace
	 */
	while (count && isspace(*s)) {
		--count;
		++s;
	}
	/*
	 * Do one full pass to verify all parameters,
	 * then do another to actually write the new settings.
	 */
	do {
		char *p = s;
		n = count;
		while (n > 0) {
			unsigned val;
			char *q = p;

			while (n > 0 && *p != ':') {
				--n;
				p++;
			}
			if (*p != ':')
				goto parse_error;
			if (p - q > MAX_LEN)
				goto parse_error;
			memcpy(name, q, p - q);
			name[p - q] = 0;

			if (n > 0) {
				--n;
				p++;
			} else
				goto parse_error;

			val = simple_strtoul(p, &q, 10);
			n -= q - p;
			p = q;
			if (n > 0 && !isspace(*p))
				goto parse_error;
			while (n > 0 && isspace(*p)) {
				--n;
				++p;
			}

377
			mutex_lock(&ide_setting_mtx);
378 379
			/* generic settings first, then driver specific ones */
			setting = ide_find_setting(ide_generic_settings, name);
380
			if (!setting) {
381 382 383 384 385 386 387 388 389 390 391
				if (drive->settings)
					setting = ide_find_setting(drive->settings, name);
				if (!setting) {
					mutex_unlock(&ide_setting_mtx);
					goto parse_error;
				}
			}
			if (for_real) {
				mul_factor = setting->mulf ? setting->mulf(drive) : 1;
				div_factor = setting->divf ? setting->divf(drive) : 1;
				ide_write_setting(drive, setting, val * div_factor / mul_factor);
L
Linus Torvalds 已提交
392
			}
393
			mutex_unlock(&ide_setting_mtx);
L
Linus Torvalds 已提交
394 395 396 397 398 399 400 401 402 403 404 405 406
		}
	} while (!for_real++);
	free_page((unsigned long)buf);
	return count;
parse_error:
	free_page((unsigned long)buf);
	printk("proc_ide_write_settings(): parse error\n");
	return -EINVAL;
}

int proc_ide_read_capacity
	(char *page, char **start, off_t off, int count, int *eof, void *data)
{
407 408
	int len = sprintf(page, "%llu\n", (long long)0x7fffffff);
	PROC_IDE_READ_RETURN(page, start, off, count, eof, len);
L
Linus Torvalds 已提交
409 410 411 412 413 414 415 416 417 418 419
}

EXPORT_SYMBOL_GPL(proc_ide_read_capacity);

int proc_ide_read_geometry
	(char *page, char **start, off_t off, int count, int *eof, void *data)
{
	ide_drive_t	*drive = (ide_drive_t *) data;
	char		*out = page;
	int		len;

420
	out += sprintf(out, "physical     %d/%d/%d\n",
L
Linus Torvalds 已提交
421
			drive->cyl, drive->head, drive->sect);
422
	out += sprintf(out, "logical      %d/%d/%d\n",
L
Linus Torvalds 已提交
423 424 425
			drive->bios_cyl, drive->bios_head, drive->bios_sect);

	len = out - page;
426
	PROC_IDE_READ_RETURN(page, start, off, count, eof, len);
L
Linus Torvalds 已提交
427 428 429 430 431 432 433 434
}

EXPORT_SYMBOL(proc_ide_read_geometry);

static int proc_ide_read_dmodel
	(char *page, char **start, off_t off, int count, int *eof, void *data)
{
	ide_drive_t	*drive = (ide_drive_t *) data;
435
	char		*m = (char *)&drive->id[ATA_ID_PROD];
L
Linus Torvalds 已提交
436 437
	int		len;

438
	len = sprintf(page, "%.40s\n", m[0] ? m : "(none)");
439
	PROC_IDE_READ_RETURN(page, start, off, count, eof, len);
L
Linus Torvalds 已提交
440 441 442 443 444 445
}

static int proc_ide_read_driver
	(char *page, char **start, off_t off, int count, int *eof, void *data)
{
	ide_drive_t	*drive = (ide_drive_t *) data;
446 447
	struct device	*dev = &drive->gendev;
	ide_driver_t	*ide_drv;
L
Linus Torvalds 已提交
448 449
	int		len;

450 451
	if (dev->driver) {
		ide_drv = container_of(dev->driver, ide_driver_t, gen_driver);
L
Linus Torvalds 已提交
452
		len = sprintf(page, "%s version %s\n",
453
				dev->driver->name, ide_drv->version);
L
Linus Torvalds 已提交
454 455
	} else
		len = sprintf(page, "ide-default version 0.9.newide\n");
456
	PROC_IDE_READ_RETURN(page, start, off, count, eof, len);
L
Linus Torvalds 已提交
457 458
}

459 460 461 462
static int ide_replace_subdriver(ide_drive_t *drive, const char *driver)
{
	struct device *dev = &drive->gendev;
	int ret = 1;
463
	int err;
464 465 466 467

	device_release_driver(dev);
	/* FIXME: device can still be in use by previous driver */
	strlcpy(drive->driver_req, driver, sizeof(drive->driver_req));
468 469 470
	err = device_attach(dev);
	if (err < 0)
		printk(KERN_WARNING "IDE: %s: device_attach error: %d\n",
471
			__func__, err);
472
	drive->driver_req[0] = 0;
473 474 475 476 477
	if (dev->driver == NULL) {
		err = device_attach(dev);
		if (err < 0)
			printk(KERN_WARNING
				"IDE: %s: device_attach(2) error: %d\n",
478
				__func__, err);
479
	}
480 481 482 483 484 485
	if (dev->driver && !strcmp(dev->driver->name, driver))
		ret = 0;

	return ret;
}

L
Linus Torvalds 已提交
486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511
static int proc_ide_write_driver
	(struct file *file, const char __user *buffer, unsigned long count, void *data)
{
	ide_drive_t	*drive = (ide_drive_t *) data;
	char name[32];

	if (!capable(CAP_SYS_ADMIN))
		return -EACCES;
	if (count > 31)
		count = 31;
	if (copy_from_user(name, buffer, count))
		return -EFAULT;
	name[count] = '\0';
	if (ide_replace_subdriver(drive, name))
		return -EINVAL;
	return count;
}

static int proc_ide_read_media
	(char *page, char **start, off_t off, int count, int *eof, void *data)
{
	ide_drive_t	*drive = (ide_drive_t *) data;
	const char	*media;
	int		len;

	switch (drive->media) {
512 513 514 515 516 517
	case ide_disk:		media = "disk\n";	break;
	case ide_cdrom:		media = "cdrom\n";	break;
	case ide_tape:		media = "tape\n";	break;
	case ide_floppy:	media = "floppy\n";	break;
	case ide_optical:	media = "optical\n";	break;
	default:		media = "UNKNOWN\n";	break;
L
Linus Torvalds 已提交
518
	}
519
	strcpy(page, media);
L
Linus Torvalds 已提交
520
	len = strlen(media);
521
	PROC_IDE_READ_RETURN(page, start, off, count, eof, len);
L
Linus Torvalds 已提交
522 523 524
}

static ide_proc_entry_t generic_drive_entries[] = {
525 526 527 528 529 530 531
	{ "driver",	S_IFREG|S_IRUGO,	 proc_ide_read_driver,
						 proc_ide_write_driver },
	{ "identify",	S_IFREG|S_IRUSR,	 proc_ide_read_identify, NULL },
	{ "media",	S_IFREG|S_IRUGO,	 proc_ide_read_media,    NULL },
	{ "model",	S_IFREG|S_IRUGO,	 proc_ide_read_dmodel,   NULL },
	{ "settings",	S_IFREG|S_IRUSR|S_IWUSR, proc_ide_read_settings,
						 proc_ide_write_settings },
L
Linus Torvalds 已提交
532 533 534
	{ NULL,	0, NULL, NULL }
};

535
static void ide_add_proc_entries(struct proc_dir_entry *dir, ide_proc_entry_t *p, void *data)
L
Linus Torvalds 已提交
536 537 538 539 540 541 542 543 544 545 546 547 548 549 550
{
	struct proc_dir_entry *ent;

	if (!dir || !p)
		return;
	while (p->name != NULL) {
		ent = create_proc_entry(p->name, p->mode, dir);
		if (!ent) return;
		ent->data = data;
		ent->read_proc = p->read_proc;
		ent->write_proc = p->write_proc;
		p++;
	}
}

551
static void ide_remove_proc_entries(struct proc_dir_entry *dir, ide_proc_entry_t *p)
L
Linus Torvalds 已提交
552 553 554 555 556 557 558 559 560
{
	if (!dir || !p)
		return;
	while (p->name != NULL) {
		remove_proc_entry(p->name, dir);
		p++;
	}
}

561 562
void ide_proc_register_driver(ide_drive_t *drive, ide_driver_t *driver)
{
563
	mutex_lock(&ide_setting_mtx);
564
	drive->settings = driver->proc_devsets(drive);
565 566
	mutex_unlock(&ide_setting_mtx);

567
	ide_add_proc_entries(drive->proc, driver->proc_entries(drive), drive);
568 569 570 571 572 573 574 575 576 577 578 579
}

EXPORT_SYMBOL(ide_proc_register_driver);

/**
 *	ide_proc_unregister_driver	-	remove driver specific data
 *	@drive: drive
 *	@driver: driver
 *
 *	Clean up the driver specific /proc files and IDE settings
 *	for a given drive.
 *
580
 *	Takes ide_setting_mtx.
581 582 583 584
 */

void ide_proc_unregister_driver(ide_drive_t *drive, ide_driver_t *driver)
{
585
	ide_remove_proc_entries(drive->proc, driver->proc_entries(drive));
586

587
	mutex_lock(&ide_setting_mtx);
588
	/*
589 590
	 * ide_setting_mtx protects both the settings list and the use
	 * of settings (we cannot take a setting out that is being used).
591
	 */
592
	drive->settings = NULL;
593
	mutex_unlock(&ide_setting_mtx);
594 595 596
}
EXPORT_SYMBOL(ide_proc_unregister_driver);

597
void ide_proc_port_register_devices(ide_hwif_t *hwif)
L
Linus Torvalds 已提交
598 599 600 601 602 603 604 605 606
{
	int	d;
	struct proc_dir_entry *ent;
	struct proc_dir_entry *parent = hwif->proc;
	char name[64];

	for (d = 0; d < MAX_DRIVES; d++) {
		ide_drive_t *drive = &hwif->drives[d];

B
Bartlomiej Zolnierkiewicz 已提交
607
		if ((drive->dev_flags & IDE_DFLAG_PRESENT) == 0 || drive->proc)
L
Linus Torvalds 已提交
608 609 610 611 612
			continue;

		drive->proc = proc_mkdir(drive->name, parent);
		if (drive->proc)
			ide_add_proc_entries(drive->proc, generic_drive_entries, drive);
613
		sprintf(name, "ide%d/%s", (drive->name[2]-'a')/2, drive->name);
L
Linus Torvalds 已提交
614 615 616 617 618
		ent = proc_symlink(drive->name, proc_ide_root, name);
		if (!ent) return;
	}
}

619
void ide_proc_unregister_device(ide_drive_t *drive)
L
Linus Torvalds 已提交
620 621 622 623
{
	if (drive->proc) {
		ide_remove_proc_entries(drive->proc, generic_drive_entries);
		remove_proc_entry(drive->name, proc_ide_root);
624
		remove_proc_entry(drive->name, drive->hwif->proc);
L
Linus Torvalds 已提交
625 626 627 628 629 630 631 632 633 634 635
		drive->proc = NULL;
	}
}

static ide_proc_entry_t hwif_entries[] = {
	{ "channel",	S_IFREG|S_IRUGO,	proc_ide_read_channel,	NULL },
	{ "mate",	S_IFREG|S_IRUGO,	proc_ide_read_mate,	NULL },
	{ "model",	S_IFREG|S_IRUGO,	proc_ide_read_imodel,	NULL },
	{ NULL,	0, NULL, NULL }
};

636
void ide_proc_register_port(ide_hwif_t *hwif)
L
Linus Torvalds 已提交
637
{
638 639
	if (!hwif->proc) {
		hwif->proc = proc_mkdir(hwif->name, proc_ide_root);
L
Linus Torvalds 已提交
640

641 642 643 644
		if (!hwif->proc)
			return;

		ide_add_proc_entries(hwif->proc, hwif_entries, hwif);
L
Linus Torvalds 已提交
645 646 647
	}
}

648
void ide_proc_unregister_port(ide_hwif_t *hwif)
L
Linus Torvalds 已提交
649 650 651 652 653 654 655 656
{
	if (hwif->proc) {
		ide_remove_proc_entries(hwif->proc, hwif_entries);
		remove_proc_entry(hwif->name, proc_ide_root);
		hwif->proc = NULL;
	}
}

657 658 659 660 661 662 663 664 665 666 667 668
static int proc_print_driver(struct device_driver *drv, void *data)
{
	ide_driver_t *ide_drv = container_of(drv, ide_driver_t, gen_driver);
	struct seq_file *s = data;

	seq_printf(s, "%s version %s\n", drv->name, ide_drv->version);

	return 0;
}

static int ide_drivers_show(struct seq_file *s, void *p)
{
669 670 671 672 673
	int err;

	err = bus_for_each_drv(&ide_bus_type, NULL, s, proc_print_driver);
	if (err < 0)
		printk(KERN_WARNING "IDE: %s: bus_for_each_drv error: %d\n",
674
			__func__, err);
675 676 677
	return 0;
}

L
Linus Torvalds 已提交
678 679
static int ide_drivers_open(struct inode *inode, struct file *file)
{
680
	return single_open(file, &ide_drivers_show, NULL);
L
Linus Torvalds 已提交
681
}
682

683
static const struct file_operations ide_drivers_operations = {
684
	.owner		= THIS_MODULE,
L
Linus Torvalds 已提交
685 686 687
	.open		= ide_drivers_open,
	.read		= seq_read,
	.llseek		= seq_lseek,
688
	.release	= single_release,
L
Linus Torvalds 已提交
689 690 691 692
};

void proc_ide_create(void)
{
693 694
	proc_ide_root = proc_mkdir("ide", NULL);

L
Linus Torvalds 已提交
695 696 697
	if (!proc_ide_root)
		return;

698
	proc_create("drivers", 0, proc_ide_root, &ide_drivers_operations);
L
Linus Torvalds 已提交
699 700 701 702
}

void proc_ide_destroy(void)
{
Z
Zhang, Yanmin 已提交
703
	remove_proc_entry("drivers", proc_ide_root);
L
Linus Torvalds 已提交
704 705
	remove_proc_entry("ide", NULL);
}