mtdpart.c 15.3 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9
/*
 * Simple MTD partitioning layer
 *
 * (C) 2000 Nicolas Pitre <nico@cam.org>
 *
 * This code is GPL
 *
 * 	02-21-2002	Thomas Gleixner <gleixner@autronix.de>
 *			added support for read_oob, write_oob
10
 */
L
Linus Torvalds 已提交
11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40

#include <linux/module.h>
#include <linux/types.h>
#include <linux/kernel.h>
#include <linux/slab.h>
#include <linux/list.h>
#include <linux/kmod.h>
#include <linux/mtd/mtd.h>
#include <linux/mtd/partitions.h>
#include <linux/mtd/compatmac.h>

/* Our partition linked list */
static LIST_HEAD(mtd_partitions);

/* Our partition node structure */
struct mtd_part {
	struct mtd_info mtd;
	struct mtd_info *master;
	u_int32_t offset;
	int index;
	struct list_head list;
	int registered;
};

/*
 * Given a pointer to the MTD object in the mtd_part structure, we can retrieve
 * the pointer to that structure with this macro.
 */
#define PART(x)  ((struct mtd_part *)(x))

41 42

/*
L
Linus Torvalds 已提交
43 44 45 46
 * MTD methods which simply translate the effective address and pass through
 * to the _real_ device.
 */

47 48
static int part_read(struct mtd_info *mtd, loff_t from, size_t len,
		size_t *retlen, u_char *buf)
L
Linus Torvalds 已提交
49 50
{
	struct mtd_part *part = PART(mtd);
51 52
	int res;

L
Linus Torvalds 已提交
53 54 55 56
	if (from >= mtd->size)
		len = 0;
	else if (from + len > mtd->size)
		len = mtd->size - from;
57
	res = part->master->read(part->master, from + part->offset,
58
				   len, retlen, buf);
59 60 61 62 63 64 65
	if (unlikely(res)) {
		if (res == -EUCLEAN)
			mtd->ecc_stats.corrected++;
		if (res == -EBADMSG)
			mtd->ecc_stats.failed++;
	}
	return res;
L
Linus Torvalds 已提交
66 67
}

68 69
static int part_point(struct mtd_info *mtd, loff_t from, size_t len,
		size_t *retlen, void **virt, resource_size_t *phys)
L
Linus Torvalds 已提交
70 71 72 73 74 75
{
	struct mtd_part *part = PART(mtd);
	if (from >= mtd->size)
		len = 0;
	else if (from + len > mtd->size)
		len = mtd->size - from;
76
	return part->master->point (part->master, from + part->offset,
77
				    len, retlen, virt, phys);
L
Linus Torvalds 已提交
78
}
79

80
static void part_unpoint(struct mtd_info *mtd, loff_t from, size_t len)
L
Linus Torvalds 已提交
81 82 83
{
	struct mtd_part *part = PART(mtd);

84
	part->master->unpoint(part->master, from + part->offset, len);
L
Linus Torvalds 已提交
85 86
}

87
static int part_read_oob(struct mtd_info *mtd, loff_t from,
88
		struct mtd_oob_ops *ops)
L
Linus Torvalds 已提交
89 90
{
	struct mtd_part *part = PART(mtd);
91
	int res;
92

L
Linus Torvalds 已提交
93
	if (from >= mtd->size)
94
		return -EINVAL;
95
	if (ops->datbuf && from + ops->len > mtd->size)
96
		return -EINVAL;
97 98 99 100 101 102 103 104 105
	res = part->master->read_oob(part->master, from + part->offset, ops);

	if (unlikely(res)) {
		if (res == -EUCLEAN)
			mtd->ecc_stats.corrected++;
		if (res == -EBADMSG)
			mtd->ecc_stats.failed++;
	}
	return res;
L
Linus Torvalds 已提交
106 107
}

108 109
static int part_read_user_prot_reg(struct mtd_info *mtd, loff_t from,
		size_t len, size_t *retlen, u_char *buf)
L
Linus Torvalds 已提交
110 111
{
	struct mtd_part *part = PART(mtd);
112
	return part->master->read_user_prot_reg(part->master, from,
L
Linus Torvalds 已提交
113 114 115
					len, retlen, buf);
}

116 117
static int part_get_user_prot_info(struct mtd_info *mtd,
		struct otp_info *buf, size_t len)
118 119
{
	struct mtd_part *part = PART(mtd);
120
	return part->master->get_user_prot_info(part->master, buf, len);
121 122
}

123 124
static int part_read_fact_prot_reg(struct mtd_info *mtd, loff_t from,
		size_t len, size_t *retlen, u_char *buf)
L
Linus Torvalds 已提交
125 126
{
	struct mtd_part *part = PART(mtd);
127
	return part->master->read_fact_prot_reg(part->master, from,
L
Linus Torvalds 已提交
128 129 130
					len, retlen, buf);
}

131 132
static int part_get_fact_prot_info(struct mtd_info *mtd, struct otp_info *buf,
		size_t len)
133 134
{
	struct mtd_part *part = PART(mtd);
135
	return part->master->get_fact_prot_info(part->master, buf, len);
136 137
}

138 139
static int part_write(struct mtd_info *mtd, loff_t to, size_t len,
		size_t *retlen, const u_char *buf)
L
Linus Torvalds 已提交
140 141 142 143 144 145 146 147
{
	struct mtd_part *part = PART(mtd);
	if (!(mtd->flags & MTD_WRITEABLE))
		return -EROFS;
	if (to >= mtd->size)
		len = 0;
	else if (to + len > mtd->size)
		len = mtd->size - to;
148
	return part->master->write(part->master, to + part->offset,
149
				    len, retlen, buf);
L
Linus Torvalds 已提交
150 151
}

152 153
static int part_panic_write(struct mtd_info *mtd, loff_t to, size_t len,
		size_t *retlen, const u_char *buf)
154 155 156 157 158 159 160 161
{
	struct mtd_part *part = PART(mtd);
	if (!(mtd->flags & MTD_WRITEABLE))
		return -EROFS;
	if (to >= mtd->size)
		len = 0;
	else if (to + len > mtd->size)
		len = mtd->size - to;
162
	return part->master->panic_write(part->master, to + part->offset,
163 164 165
				    len, retlen, buf);
}

166
static int part_write_oob(struct mtd_info *mtd, loff_t to,
167
		struct mtd_oob_ops *ops)
L
Linus Torvalds 已提交
168 169
{
	struct mtd_part *part = PART(mtd);
170

L
Linus Torvalds 已提交
171 172
	if (!(mtd->flags & MTD_WRITEABLE))
		return -EROFS;
173

L
Linus Torvalds 已提交
174
	if (to >= mtd->size)
175
		return -EINVAL;
176
	if (ops->datbuf && to + ops->len > mtd->size)
177 178
		return -EINVAL;
	return part->master->write_oob(part->master, to + part->offset, ops);
L
Linus Torvalds 已提交
179 180
}

181 182
static int part_write_user_prot_reg(struct mtd_info *mtd, loff_t from,
		size_t len, size_t *retlen, u_char *buf)
L
Linus Torvalds 已提交
183 184
{
	struct mtd_part *part = PART(mtd);
185
	return part->master->write_user_prot_reg(part->master, from,
L
Linus Torvalds 已提交
186 187 188
					len, retlen, buf);
}

189 190
static int part_lock_user_prot_reg(struct mtd_info *mtd, loff_t from,
		size_t len)
191 192
{
	struct mtd_part *part = PART(mtd);
193
	return part->master->lock_user_prot_reg(part->master, from, len);
194 195
}

196 197
static int part_writev(struct mtd_info *mtd, const struct kvec *vecs,
		unsigned long count, loff_t to, size_t *retlen)
L
Linus Torvalds 已提交
198 199 200 201
{
	struct mtd_part *part = PART(mtd);
	if (!(mtd->flags & MTD_WRITEABLE))
		return -EROFS;
202
	return part->master->writev(part->master, vecs, count,
L
Linus Torvalds 已提交
203 204 205
					to + part->offset, retlen);
}

206
static int part_erase(struct mtd_info *mtd, struct erase_info *instr)
L
Linus Torvalds 已提交
207 208 209 210 211 212 213 214 215
{
	struct mtd_part *part = PART(mtd);
	int ret;
	if (!(mtd->flags & MTD_WRITEABLE))
		return -EROFS;
	if (instr->addr >= mtd->size)
		return -EINVAL;
	instr->addr += part->offset;
	ret = part->master->erase(part->master, instr);
216 217 218 219 220
	if (ret) {
		if (instr->fail_addr != 0xffffffff)
			instr->fail_addr -= part->offset;
		instr->addr -= part->offset;
	}
L
Linus Torvalds 已提交
221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237
	return ret;
}

void mtd_erase_callback(struct erase_info *instr)
{
	if (instr->mtd->erase == part_erase) {
		struct mtd_part *part = PART(instr->mtd);

		if (instr->fail_addr != 0xffffffff)
			instr->fail_addr -= part->offset;
		instr->addr -= part->offset;
	}
	if (instr->callback)
		instr->callback(instr);
}
EXPORT_SYMBOL_GPL(mtd_erase_callback);

238
static int part_lock(struct mtd_info *mtd, loff_t ofs, size_t len)
L
Linus Torvalds 已提交
239 240
{
	struct mtd_part *part = PART(mtd);
241
	if ((len + ofs) > mtd->size)
L
Linus Torvalds 已提交
242 243 244 245
		return -EINVAL;
	return part->master->lock(part->master, ofs + part->offset, len);
}

246
static int part_unlock(struct mtd_info *mtd, loff_t ofs, size_t len)
L
Linus Torvalds 已提交
247 248
{
	struct mtd_part *part = PART(mtd);
249
	if ((len + ofs) > mtd->size)
L
Linus Torvalds 已提交
250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271
		return -EINVAL;
	return part->master->unlock(part->master, ofs + part->offset, len);
}

static void part_sync(struct mtd_info *mtd)
{
	struct mtd_part *part = PART(mtd);
	part->master->sync(part->master);
}

static int part_suspend(struct mtd_info *mtd)
{
	struct mtd_part *part = PART(mtd);
	return part->master->suspend(part->master);
}

static void part_resume(struct mtd_info *mtd)
{
	struct mtd_part *part = PART(mtd);
	part->master->resume(part->master);
}

272
static int part_block_isbad(struct mtd_info *mtd, loff_t ofs)
L
Linus Torvalds 已提交
273 274 275 276 277 278 279 280
{
	struct mtd_part *part = PART(mtd);
	if (ofs >= mtd->size)
		return -EINVAL;
	ofs += part->offset;
	return part->master->block_isbad(part->master, ofs);
}

281
static int part_block_markbad(struct mtd_info *mtd, loff_t ofs)
L
Linus Torvalds 已提交
282 283
{
	struct mtd_part *part = PART(mtd);
284 285
	int res;

L
Linus Torvalds 已提交
286 287 288 289 290
	if (!(mtd->flags & MTD_WRITEABLE))
		return -EROFS;
	if (ofs >= mtd->size)
		return -EINVAL;
	ofs += part->offset;
291 292 293 294
	res = part->master->block_markbad(part->master, ofs);
	if (!res)
		mtd->ecc_stats.badblocks++;
	return res;
L
Linus Torvalds 已提交
295 296
}

297 298
/*
 * This function unregisters and destroy all slave MTD objects which are
L
Linus Torvalds 已提交
299 300 301 302 303
 * attached to the given master MTD object.
 */

int del_mtd_partitions(struct mtd_info *master)
{
304
	struct mtd_part *slave, *next;
L
Linus Torvalds 已提交
305

306
	list_for_each_entry_safe(slave, next, &mtd_partitions, list)
L
Linus Torvalds 已提交
307
		if (slave->master == master) {
308
			list_del(&slave->list);
309
			if (slave->registered)
L
Linus Torvalds 已提交
310 311 312 313 314 315
				del_mtd_device(&slave->mtd);
			kfree(slave);
		}

	return 0;
}
316
EXPORT_SYMBOL(del_mtd_partitions);
L
Linus Torvalds 已提交
317

318 319 320 321 322 323 324
static struct mtd_part *add_one_partition(struct mtd_info *master,
		const struct mtd_partition *part, int partno,
		u_int32_t cur_offset)
{
	struct mtd_part *slave;

	/* allocate the partition structure */
325
	slave = kzalloc(sizeof(*slave), GFP_KERNEL);
326
	if (!slave) {
327
		printk(KERN_ERR"memory allocation error while creating partitions for \"%s\"\n",
328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351
			master->name);
		del_mtd_partitions(master);
		return NULL;
	}
	list_add(&slave->list, &mtd_partitions);

	/* set up the MTD object for this partition */
	slave->mtd.type = master->type;
	slave->mtd.flags = master->flags & ~part->mask_flags;
	slave->mtd.size = part->size;
	slave->mtd.writesize = master->writesize;
	slave->mtd.oobsize = master->oobsize;
	slave->mtd.oobavail = master->oobavail;
	slave->mtd.subpage_sft = master->subpage_sft;

	slave->mtd.name = part->name;
	slave->mtd.owner = master->owner;

	slave->mtd.read = part_read;
	slave->mtd.write = part_write;

	if (master->panic_write)
		slave->mtd.panic_write = part_panic_write;

352
	if (master->point && master->unpoint) {
353 354 355 356 357 358 359 360
		slave->mtd.point = part_point;
		slave->mtd.unpoint = part_unpoint;
	}

	if (master->read_oob)
		slave->mtd.read_oob = part_read_oob;
	if (master->write_oob)
		slave->mtd.write_oob = part_write_oob;
361
	if (master->read_user_prot_reg)
362
		slave->mtd.read_user_prot_reg = part_read_user_prot_reg;
363
	if (master->read_fact_prot_reg)
364
		slave->mtd.read_fact_prot_reg = part_read_fact_prot_reg;
365
	if (master->write_user_prot_reg)
366
		slave->mtd.write_user_prot_reg = part_write_user_prot_reg;
367
	if (master->lock_user_prot_reg)
368
		slave->mtd.lock_user_prot_reg = part_lock_user_prot_reg;
369
	if (master->get_user_prot_info)
370
		slave->mtd.get_user_prot_info = part_get_user_prot_info;
371
	if (master->get_fact_prot_info)
372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408
		slave->mtd.get_fact_prot_info = part_get_fact_prot_info;
	if (master->sync)
		slave->mtd.sync = part_sync;
	if (!partno && master->suspend && master->resume) {
			slave->mtd.suspend = part_suspend;
			slave->mtd.resume = part_resume;
	}
	if (master->writev)
		slave->mtd.writev = part_writev;
	if (master->lock)
		slave->mtd.lock = part_lock;
	if (master->unlock)
		slave->mtd.unlock = part_unlock;
	if (master->block_isbad)
		slave->mtd.block_isbad = part_block_isbad;
	if (master->block_markbad)
		slave->mtd.block_markbad = part_block_markbad;
	slave->mtd.erase = part_erase;
	slave->master = master;
	slave->offset = part->offset;
	slave->index = partno;

	if (slave->offset == MTDPART_OFS_APPEND)
		slave->offset = cur_offset;
	if (slave->offset == MTDPART_OFS_NXTBLK) {
		slave->offset = cur_offset;
		if ((cur_offset % master->erasesize) != 0) {
			/* Round up to next erasesize */
			slave->offset = ((cur_offset / master->erasesize) + 1) * master->erasesize;
			printk(KERN_NOTICE "Moving partition %d: "
			       "0x%08x -> 0x%08x\n", partno,
			       cur_offset, slave->offset);
		}
	}
	if (slave->mtd.size == MTDPART_SIZ_FULL)
		slave->mtd.size = master->size - slave->offset;

409
	printk(KERN_NOTICE "0x%08x-0x%08x : \"%s\"\n", slave->offset,
410 411 412 413 414 415 416
		slave->offset + slave->mtd.size, slave->mtd.name);

	/* let's do some sanity checks */
	if (slave->offset >= master->size) {
			/* let's register it anyway to preserve ordering */
		slave->offset = 0;
		slave->mtd.size = 0;
417
		printk(KERN_ERR"mtd: partition \"%s\" is out of reach -- disabled\n",
418 419 420 421
			part->name);
	}
	if (slave->offset + slave->mtd.size > master->size) {
		slave->mtd.size = master->size - slave->offset;
422
		printk(KERN_WARNING"mtd: partition \"%s\" extends beyond the end of device \"%s\" -- size truncated to %#x\n",
423 424
			part->name, master->name, slave->mtd.size);
	}
425
	if (master->numeraseregions > 1) {
426
		/* Deal with variable erase size stuff */
427 428
		int i, max = master->numeraseregions;
		u32 end = slave->offset + slave->mtd.size;
429 430
		struct mtd_erase_region_info *regions = master->eraseregions;

431 432 433
		/* Find the first erase regions which is part of this
		 * partition. */
		for (i = 0; i < max && regions[i].offset <= slave->offset; i++)
434
			;
435 436
		/* The loop searched for the region _behind_ the first one */
		i--;
437

438 439
		/* Pick biggest erasesize */
		for (; i < max && regions[i].offset < end; i++) {
440 441 442 443
			if (slave->mtd.erasesize < regions[i].erasesize) {
				slave->mtd.erasesize = regions[i].erasesize;
			}
		}
444
		BUG_ON(slave->mtd.erasesize == 0);
445 446 447 448 449 450 451 452
	} else {
		/* Single erase size */
		slave->mtd.erasesize = master->erasesize;
	}

	if ((slave->mtd.flags & MTD_WRITEABLE) &&
	    (slave->offset % slave->mtd.erasesize)) {
		/* Doesn't start on a boundary of major erase size */
453 454
		/* FIXME: Let it be writable if it is on a boundary of
		 * _minor_ erase size though */
455
		slave->mtd.flags &= ~MTD_WRITEABLE;
456
		printk(KERN_WARNING"mtd: partition \"%s\" doesn't start on an erase block boundary -- force read-only\n",
457 458 459 460 461
			part->name);
	}
	if ((slave->mtd.flags & MTD_WRITEABLE) &&
	    (slave->mtd.size % slave->mtd.erasesize)) {
		slave->mtd.flags &= ~MTD_WRITEABLE;
462
		printk(KERN_WARNING"mtd: partition \"%s\" doesn't end on an erase block -- force read-only\n",
463 464 465 466 467 468 469
			part->name);
	}

	slave->mtd.ecclayout = master->ecclayout;
	if (master->block_isbad) {
		uint32_t offs = 0;

470
		while (offs < slave->mtd.size) {
471 472 473 474 475 476 477
			if (master->block_isbad(master,
						offs + slave->offset))
				slave->mtd.ecc_stats.badblocks++;
			offs += slave->mtd.erasesize;
		}
	}

478 479
	if (part->mtdp) {
		/* store the object pointer (caller may or may not register it*/
480 481 482 483 484 485 486 487 488 489
		*part->mtdp = &slave->mtd;
		slave->registered = 0;
	} else {
		/* register our partition */
		add_mtd_device(&slave->mtd);
		slave->registered = 1;
	}
	return slave;
}

L
Linus Torvalds 已提交
490 491 492 493 494 495 496
/*
 * This function, given a master MTD object and a partition table, creates
 * and registers slave MTD objects which are bound to the master according to
 * the partition definitions.
 * (Q: should we register the master MTD object as well?)
 */

497
int add_mtd_partitions(struct mtd_info *master,
L
Linus Torvalds 已提交
498 499 500 501 502 503 504
		       const struct mtd_partition *parts,
		       int nbparts)
{
	struct mtd_part *slave;
	u_int32_t cur_offset = 0;
	int i;

505
	printk(KERN_NOTICE "Creating %d MTD partitions on \"%s\":\n", nbparts, master->name);
L
Linus Torvalds 已提交
506 507

	for (i = 0; i < nbparts; i++) {
508 509
		slave = add_one_partition(master, parts + i, i, cur_offset);
		if (!slave)
L
Linus Torvalds 已提交
510 511 512 513 514 515 516 517 518 519 520 521 522
			return -ENOMEM;
		cur_offset = slave->offset + slave->mtd.size;
	}

	return 0;
}
EXPORT_SYMBOL(add_mtd_partitions);

static DEFINE_SPINLOCK(part_parser_lock);
static LIST_HEAD(part_parsers);

static struct mtd_part_parser *get_partition_parser(const char *name)
{
523
	struct mtd_part_parser *p, *ret = NULL;
L
Linus Torvalds 已提交
524

525
	spin_lock(&part_parser_lock);
L
Linus Torvalds 已提交
526

527
	list_for_each_entry(p, &part_parsers, list)
L
Linus Torvalds 已提交
528 529 530 531
		if (!strcmp(p->name, name) && try_module_get(p->owner)) {
			ret = p;
			break;
		}
532

L
Linus Torvalds 已提交
533 534 535 536 537 538 539 540 541 542 543 544 545
	spin_unlock(&part_parser_lock);

	return ret;
}

int register_mtd_parser(struct mtd_part_parser *p)
{
	spin_lock(&part_parser_lock);
	list_add(&p->list, &part_parsers);
	spin_unlock(&part_parser_lock);

	return 0;
}
546
EXPORT_SYMBOL_GPL(register_mtd_parser);
L
Linus Torvalds 已提交
547 548 549 550 551 552 553 554

int deregister_mtd_parser(struct mtd_part_parser *p)
{
	spin_lock(&part_parser_lock);
	list_del(&p->list);
	spin_unlock(&part_parser_lock);
	return 0;
}
555
EXPORT_SYMBOL_GPL(deregister_mtd_parser);
L
Linus Torvalds 已提交
556

557
int parse_mtd_partitions(struct mtd_info *master, const char **types,
L
Linus Torvalds 已提交
558 559 560 561
			 struct mtd_partition **pparts, unsigned long origin)
{
	struct mtd_part_parser *parser;
	int ret = 0;
562

L
Linus Torvalds 已提交
563 564 565 566 567 568 569 570 571 572 573 574 575
	for ( ; ret <= 0 && *types; types++) {
		parser = get_partition_parser(*types);
#ifdef CONFIG_KMOD
		if (!parser && !request_module("%s", *types))
				parser = get_partition_parser(*types);
#endif
		if (!parser) {
			printk(KERN_NOTICE "%s partition parsing not available\n",
			       *types);
			continue;
		}
		ret = (*parser->parse_fn)(master, pparts, origin);
		if (ret > 0) {
576
			printk(KERN_NOTICE "%d %s partitions found on MTD device %s\n",
L
Linus Torvalds 已提交
577 578 579 580 581 582 583
			       ret, parser->name, master->name);
		}
		put_partition_parser(parser);
	}
	return ret;
}
EXPORT_SYMBOL_GPL(parse_mtd_partitions);