mtdpart.c 15.1 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 427 428 429 430
		/* Deal with variable erase size stuff */
		int i;
		struct mtd_erase_region_info *regions = master->eraseregions;

		/* Find the first erase regions which is part of this partition. */
431
		for (i = 0; i < master->numeraseregions && regions[i].offset <= slave->offset; i++)
432 433 434 435 436 437 438 439 440 441 442 443 444 445 446
			;

		for (i--; i < master->numeraseregions && regions[i].offset < slave->offset + slave->mtd.size; i++) {
			if (slave->mtd.erasesize < regions[i].erasesize) {
				slave->mtd.erasesize = regions[i].erasesize;
			}
		}
	} 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 */
447 448
		/* FIXME: Let it be writable if it is on a boundary of
		 * _minor_ erase size though */
449
		slave->mtd.flags &= ~MTD_WRITEABLE;
450
		printk(KERN_WARNING"mtd: partition \"%s\" doesn't start on an erase block boundary -- force read-only\n",
451 452 453 454 455
			part->name);
	}
	if ((slave->mtd.flags & MTD_WRITEABLE) &&
	    (slave->mtd.size % slave->mtd.erasesize)) {
		slave->mtd.flags &= ~MTD_WRITEABLE;
456
		printk(KERN_WARNING"mtd: partition \"%s\" doesn't end on an erase block -- force read-only\n",
457 458 459 460 461 462 463
			part->name);
	}

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

464
		while (offs < slave->mtd.size) {
465 466 467 468 469 470 471
			if (master->block_isbad(master,
						offs + slave->offset))
				slave->mtd.ecc_stats.badblocks++;
			offs += slave->mtd.erasesize;
		}
	}

472 473
	if (part->mtdp) {
		/* store the object pointer (caller may or may not register it*/
474 475 476 477 478 479 480 481 482 483
		*part->mtdp = &slave->mtd;
		slave->registered = 0;
	} else {
		/* register our partition */
		add_mtd_device(&slave->mtd);
		slave->registered = 1;
	}
	return slave;
}

L
Linus Torvalds 已提交
484 485 486 487 488 489 490
/*
 * 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?)
 */

491
int add_mtd_partitions(struct mtd_info *master,
L
Linus Torvalds 已提交
492 493 494 495 496 497 498
		       const struct mtd_partition *parts,
		       int nbparts)
{
	struct mtd_part *slave;
	u_int32_t cur_offset = 0;
	int i;

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

	for (i = 0; i < nbparts; i++) {
502 503
		slave = add_one_partition(master, parts + i, i, cur_offset);
		if (!slave)
L
Linus Torvalds 已提交
504 505 506 507 508 509 510 511 512 513 514 515 516
			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)
{
517
	struct mtd_part_parser *p, *ret = NULL;
L
Linus Torvalds 已提交
518

519
	spin_lock(&part_parser_lock);
L
Linus Torvalds 已提交
520

521
	list_for_each_entry(p, &part_parsers, list)
L
Linus Torvalds 已提交
522 523 524 525
		if (!strcmp(p->name, name) && try_module_get(p->owner)) {
			ret = p;
			break;
		}
526

L
Linus Torvalds 已提交
527 528 529 530 531 532 533 534 535 536 537 538 539
	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;
}
540
EXPORT_SYMBOL_GPL(register_mtd_parser);
L
Linus Torvalds 已提交
541 542 543 544 545 546 547 548

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;
}
549
EXPORT_SYMBOL_GPL(deregister_mtd_parser);
L
Linus Torvalds 已提交
550

551
int parse_mtd_partitions(struct mtd_info *master, const char **types,
L
Linus Torvalds 已提交
552 553 554 555
			 struct mtd_partition **pparts, unsigned long origin)
{
	struct mtd_part_parser *parser;
	int ret = 0;
556

L
Linus Torvalds 已提交
557 558 559 560 561 562 563 564 565 566 567 568 569
	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) {
570
			printk(KERN_NOTICE "%d %s partitions found on MTD device %s\n",
L
Linus Torvalds 已提交
571 572 573 574 575 576 577
			       ret, parser->name, master->name);
		}
		put_partition_parser(parser);
	}
	return ret;
}
EXPORT_SYMBOL_GPL(parse_mtd_partitions);