2005-04-17 02:20:36 +04:00
/*
* Simple MTD partitioning layer
*
2010-08-08 23:58:20 +04:00
* Copyright © 2000 Nicolas Pitre < nico @ fluxnic . net >
* Copyright © 2002 Thomas Gleixner < gleixner @ linutronix . de >
* Copyright © 2000 - 2010 David Woodhouse < dwmw2 @ infradead . org >
2005-04-17 02:20:36 +04:00
*
2010-08-08 23:58:20 +04:00
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin St , Fifth Floor , Boston , MA 02110 - 1301 USA
2005-04-17 02:20:36 +04:00
*
2005-11-07 14:15:26 +03:00
*/
2005-04-17 02:20:36 +04:00
# 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>
2010-09-17 14:31:41 +04:00
# include <linux/err.h>
2005-04-17 02:20:36 +04:00
/* Our partition linked list */
static LIST_HEAD ( mtd_partitions ) ;
2010-09-17 14:31:41 +04:00
static DEFINE_MUTEX ( mtd_partitions_mutex ) ;
2005-04-17 02:20:36 +04:00
/* Our partition node structure */
struct mtd_part {
struct mtd_info mtd ;
struct mtd_info * master ;
2008-12-10 16:37:21 +03:00
uint64_t offset ;
2005-04-17 02:20:36 +04:00
struct list_head list ;
} ;
/*
* 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))
2005-11-07 14:15:26 +03:00
/*
2005-04-17 02:20:36 +04:00
* MTD methods which simply translate the effective address and pass through
* to the _real_ device .
*/
2008-07-18 20:00:33 +04:00
static int part_read ( struct mtd_info * mtd , loff_t from , size_t len ,
size_t * retlen , u_char * buf )
2005-04-17 02:20:36 +04:00
{
struct mtd_part * part = PART ( mtd ) ;
2009-03-27 01:41:09 +03:00
struct mtd_ecc_stats stats ;
2006-05-30 02:37:34 +04:00
int res ;
2009-03-27 01:41:09 +03:00
stats = part - > master - > ecc_stats ;
2005-04-17 02:20:36 +04:00
if ( from > = mtd - > size )
len = 0 ;
else if ( from + len > mtd - > size )
len = mtd - > size - from ;
2008-07-18 20:00:33 +04:00
res = part - > master - > read ( part - > master , from + part - > offset ,
2006-05-23 19:21:03 +04:00
len , retlen , buf ) ;
2006-05-30 02:37:34 +04:00
if ( unlikely ( res ) ) {
if ( res = = - EUCLEAN )
2009-03-27 01:41:09 +03:00
mtd - > ecc_stats . corrected + = part - > master - > ecc_stats . corrected - stats . corrected ;
2006-05-30 02:37:34 +04:00
if ( res = = - EBADMSG )
2009-03-27 01:41:09 +03:00
mtd - > ecc_stats . failed + = part - > master - > ecc_stats . failed - stats . failed ;
2006-05-30 02:37:34 +04:00
}
return res ;
2005-04-17 02:20:36 +04:00
}
2008-07-18 20:00:33 +04:00
static int part_point ( struct mtd_info * mtd , loff_t from , size_t len ,
size_t * retlen , void * * virt , resource_size_t * phys )
2005-04-17 02:20:36 +04:00
{
struct mtd_part * part = PART ( mtd ) ;
if ( from > = mtd - > size )
len = 0 ;
else if ( from + len > mtd - > size )
len = mtd - > size - from ;
2005-11-07 14:15:26 +03:00
return part - > master - > point ( part - > master , from + part - > offset ,
2008-04-30 10:26:49 +04:00
len , retlen , virt , phys ) ;
2005-04-17 02:20:36 +04:00
}
2006-05-23 19:21:03 +04:00
2008-04-30 10:26:49 +04:00
static void part_unpoint ( struct mtd_info * mtd , loff_t from , size_t len )
2005-04-17 02:20:36 +04:00
{
struct mtd_part * part = PART ( mtd ) ;
2008-04-30 10:26:49 +04:00
part - > master - > unpoint ( part - > master , from + part - > offset , len ) ;
2005-04-17 02:20:36 +04:00
}
2009-02-12 13:40:00 +03:00
static unsigned long part_get_unmapped_area ( struct mtd_info * mtd ,
unsigned long len ,
unsigned long offset ,
unsigned long flags )
{
struct mtd_part * part = PART ( mtd ) ;
offset + = part - > offset ;
return part - > master - > get_unmapped_area ( part - > master , len , offset ,
flags ) ;
}
2006-05-29 05:26:58 +04:00
static int part_read_oob ( struct mtd_info * mtd , loff_t from ,
2008-07-18 20:00:33 +04:00
struct mtd_oob_ops * ops )
2005-04-17 02:20:36 +04:00
{
struct mtd_part * part = PART ( mtd ) ;
2006-05-30 02:37:34 +04:00
int res ;
2006-05-29 05:26:58 +04:00
2005-04-17 02:20:36 +04:00
if ( from > = mtd - > size )
2006-05-29 05:26:58 +04:00
return - EINVAL ;
2006-11-03 18:20:38 +03:00
if ( ops - > datbuf & & from + ops - > len > mtd - > size )
2006-05-29 05:26:58 +04:00
return - EINVAL ;
2006-05-30 02:37:34 +04:00
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 ;
2005-04-17 02:20:36 +04:00
}
2008-07-18 20:00:33 +04:00
static int part_read_user_prot_reg ( struct mtd_info * mtd , loff_t from ,
size_t len , size_t * retlen , u_char * buf )
2005-04-17 02:20:36 +04:00
{
struct mtd_part * part = PART ( mtd ) ;
2008-07-18 20:00:33 +04:00
return part - > master - > read_user_prot_reg ( part - > master , from ,
2005-04-17 02:20:36 +04:00
len , retlen , buf ) ;
}
2008-07-18 20:00:33 +04:00
static int part_get_user_prot_info ( struct mtd_info * mtd ,
struct otp_info * buf , size_t len )
2005-02-08 20:11:19 +03:00
{
struct mtd_part * part = PART ( mtd ) ;
2008-07-18 20:00:33 +04:00
return part - > master - > get_user_prot_info ( part - > master , buf , len ) ;
2005-02-08 20:11:19 +03:00
}
2008-07-18 20:00:33 +04:00
static int part_read_fact_prot_reg ( struct mtd_info * mtd , loff_t from ,
size_t len , size_t * retlen , u_char * buf )
2005-04-17 02:20:36 +04:00
{
struct mtd_part * part = PART ( mtd ) ;
2008-07-18 20:00:33 +04:00
return part - > master - > read_fact_prot_reg ( part - > master , from ,
2005-04-17 02:20:36 +04:00
len , retlen , buf ) ;
}
2008-07-18 20:00:33 +04:00
static int part_get_fact_prot_info ( struct mtd_info * mtd , struct otp_info * buf ,
size_t len )
2005-02-08 20:11:19 +03:00
{
struct mtd_part * part = PART ( mtd ) ;
2008-07-18 20:00:33 +04:00
return part - > master - > get_fact_prot_info ( part - > master , buf , len ) ;
2005-02-08 20:11:19 +03:00
}
2008-07-18 20:00:33 +04:00
static int part_write ( struct mtd_info * mtd , loff_t to , size_t len ,
size_t * retlen , const u_char * buf )
2005-04-17 02:20:36 +04:00
{
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 ;
2008-07-18 20:00:33 +04:00
return part - > master - > write ( part - > master , to + part - > offset ,
2006-05-23 19:21:03 +04:00
len , retlen , buf ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-18 20:00:33 +04:00
static int part_panic_write ( struct mtd_info * mtd , loff_t to , size_t len ,
size_t * retlen , const u_char * buf )
2008-02-06 13:17:15 +03:00
{
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 ;
2008-07-18 20:00:33 +04:00
return part - > master - > panic_write ( part - > master , to + part - > offset ,
2008-02-06 13:17:15 +03:00
len , retlen , buf ) ;
}
2006-05-29 05:26:58 +04:00
static int part_write_oob ( struct mtd_info * mtd , loff_t to ,
2008-07-18 20:00:33 +04:00
struct mtd_oob_ops * ops )
2005-04-17 02:20:36 +04:00
{
struct mtd_part * part = PART ( mtd ) ;
2006-05-29 05:26:58 +04:00
2005-04-17 02:20:36 +04:00
if ( ! ( mtd - > flags & MTD_WRITEABLE ) )
return - EROFS ;
2006-05-29 05:26:58 +04:00
2005-04-17 02:20:36 +04:00
if ( to > = mtd - > size )
2006-05-29 05:26:58 +04:00
return - EINVAL ;
2006-11-03 18:20:38 +03:00
if ( ops - > datbuf & & to + ops - > len > mtd - > size )
2006-05-29 05:26:58 +04:00
return - EINVAL ;
return part - > master - > write_oob ( part - > master , to + part - > offset , ops ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-18 20:00:33 +04:00
static int part_write_user_prot_reg ( struct mtd_info * mtd , loff_t from ,
size_t len , size_t * retlen , u_char * buf )
2005-04-17 02:20:36 +04:00
{
struct mtd_part * part = PART ( mtd ) ;
2008-07-18 20:00:33 +04:00
return part - > master - > write_user_prot_reg ( part - > master , from ,
2005-04-17 02:20:36 +04:00
len , retlen , buf ) ;
}
2008-07-18 20:00:33 +04:00
static int part_lock_user_prot_reg ( struct mtd_info * mtd , loff_t from ,
size_t len )
2005-02-08 20:11:19 +03:00
{
struct mtd_part * part = PART ( mtd ) ;
2008-07-18 20:00:33 +04:00
return part - > master - > lock_user_prot_reg ( part - > master , from , len ) ;
2005-02-08 20:11:19 +03:00
}
2008-07-18 20:00:33 +04:00
static int part_writev ( struct mtd_info * mtd , const struct kvec * vecs ,
unsigned long count , loff_t to , size_t * retlen )
2005-04-17 02:20:36 +04:00
{
struct mtd_part * part = PART ( mtd ) ;
if ( ! ( mtd - > flags & MTD_WRITEABLE ) )
return - EROFS ;
2008-07-18 20:00:33 +04:00
return part - > master - > writev ( part - > master , vecs , count ,
2005-04-17 02:20:36 +04:00
to + part - > offset , retlen ) ;
}
2008-07-18 20:00:33 +04:00
static int part_erase ( struct mtd_info * mtd , struct erase_info * instr )
2005-04-17 02:20:36 +04:00
{
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 ) ;
2007-03-08 13:20:12 +03:00
if ( ret ) {
2008-08-12 13:40:50 +04:00
if ( instr - > fail_addr ! = MTD_FAIL_ADDR_UNKNOWN )
2007-03-08 13:20:12 +03:00
instr - > fail_addr - = part - > offset ;
instr - > addr - = part - > offset ;
}
2005-04-17 02:20:36 +04:00
return ret ;
}
void mtd_erase_callback ( struct erase_info * instr )
{
if ( instr - > mtd - > erase = = part_erase ) {
struct mtd_part * part = PART ( instr - > mtd ) ;
2008-08-12 13:40:50 +04:00
if ( instr - > fail_addr ! = MTD_FAIL_ADDR_UNKNOWN )
2005-04-17 02:20:36 +04:00
instr - > fail_addr - = part - > offset ;
instr - > addr - = part - > offset ;
}
if ( instr - > callback )
instr - > callback ( instr ) ;
}
EXPORT_SYMBOL_GPL ( mtd_erase_callback ) ;
2008-12-10 16:37:21 +03:00
static int part_lock ( struct mtd_info * mtd , loff_t ofs , uint64_t len )
2005-04-17 02:20:36 +04:00
{
struct mtd_part * part = PART ( mtd ) ;
2005-11-07 14:15:26 +03:00
if ( ( len + ofs ) > mtd - > size )
2005-04-17 02:20:36 +04:00
return - EINVAL ;
return part - > master - > lock ( part - > master , ofs + part - > offset , len ) ;
}
2008-12-10 16:37:21 +03:00
static int part_unlock ( struct mtd_info * mtd , loff_t ofs , uint64_t len )
2005-04-17 02:20:36 +04:00
{
struct mtd_part * part = PART ( mtd ) ;
2005-11-07 14:15:26 +03:00
if ( ( len + ofs ) > mtd - > size )
2005-04-17 02:20:36 +04:00
return - EINVAL ;
return part - > master - > unlock ( part - > master , ofs + part - > offset , len ) ;
}
2010-06-14 20:10:33 +04:00
static int part_is_locked ( struct mtd_info * mtd , loff_t ofs , uint64_t len )
{
struct mtd_part * part = PART ( mtd ) ;
if ( ( len + ofs ) > mtd - > size )
return - EINVAL ;
return part - > master - > is_locked ( part - > master , ofs + part - > offset , len ) ;
}
2005-04-17 02:20:36 +04:00
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 ) ;
}
2008-07-18 20:00:33 +04:00
static int part_block_isbad ( struct mtd_info * mtd , loff_t ofs )
2005-04-17 02:20:36 +04:00
{
struct mtd_part * part = PART ( mtd ) ;
if ( ofs > = mtd - > size )
return - EINVAL ;
ofs + = part - > offset ;
return part - > master - > block_isbad ( part - > master , ofs ) ;
}
2008-07-18 20:00:33 +04:00
static int part_block_markbad ( struct mtd_info * mtd , loff_t ofs )
2005-04-17 02:20:36 +04:00
{
struct mtd_part * part = PART ( mtd ) ;
2006-05-30 02:37:34 +04:00
int res ;
2005-04-17 02:20:36 +04:00
if ( ! ( mtd - > flags & MTD_WRITEABLE ) )
return - EROFS ;
if ( ofs > = mtd - > size )
return - EINVAL ;
ofs + = part - > offset ;
2006-05-30 02:37:34 +04:00
res = part - > master - > block_markbad ( part - > master , ofs ) ;
if ( ! res )
mtd - > ecc_stats . badblocks + + ;
return res ;
2005-04-17 02:20:36 +04:00
}
2010-09-17 14:31:41 +04:00
static inline void free_partition ( struct mtd_part * p )
{
kfree ( p - > mtd . name ) ;
kfree ( p ) ;
}
2005-11-07 14:15:26 +03:00
/*
* This function unregisters and destroy all slave MTD objects which are
2005-04-17 02:20:36 +04:00
* attached to the given master MTD object .
*/
int del_mtd_partitions ( struct mtd_info * master )
{
2008-05-19 23:11:50 +04:00
struct mtd_part * slave , * next ;
2010-09-17 14:31:41 +04:00
int ret , err = 0 ;
2005-04-17 02:20:36 +04:00
2010-09-17 14:31:41 +04:00
mutex_lock ( & mtd_partitions_mutex ) ;
2008-05-19 23:11:50 +04:00
list_for_each_entry_safe ( slave , next , & mtd_partitions , list )
2005-04-17 02:20:36 +04:00
if ( slave - > master = = master ) {
2010-09-17 14:31:41 +04:00
ret = del_mtd_device ( & slave - > mtd ) ;
if ( ret < 0 ) {
err = ret ;
continue ;
}
2008-05-19 23:11:50 +04:00
list_del ( & slave - > list ) ;
2010-09-17 14:31:41 +04:00
free_partition ( slave ) ;
2005-04-17 02:20:36 +04:00
}
2010-09-17 14:31:41 +04:00
mutex_unlock ( & mtd_partitions_mutex ) ;
2005-04-17 02:20:36 +04:00
2010-09-17 14:31:41 +04:00
return err ;
2005-04-17 02:20:36 +04:00
}
2008-07-18 20:00:33 +04:00
EXPORT_SYMBOL ( del_mtd_partitions ) ;
2005-04-17 02:20:36 +04:00
2010-09-17 14:31:41 +04:00
static struct mtd_part * allocate_partition ( struct mtd_info * master ,
const struct mtd_partition * part , int partno ,
uint64_t cur_offset )
2008-07-18 20:00:18 +04:00
{
struct mtd_part * slave ;
2010-09-17 14:31:41 +04:00
char * name ;
2008-07-18 20:00:18 +04:00
/* allocate the partition structure */
2008-07-18 20:00:33 +04:00
slave = kzalloc ( sizeof ( * slave ) , GFP_KERNEL ) ;
2010-09-17 14:31:41 +04:00
name = kstrdup ( part - > name , GFP_KERNEL ) ;
if ( ! name | | ! slave ) {
2008-07-18 20:00:33 +04:00
printk ( KERN_ERR " memory allocation error while creating partitions for \" %s \" \n " ,
2010-09-17 14:31:41 +04:00
master - > name ) ;
kfree ( name ) ;
kfree ( slave ) ;
return ERR_PTR ( - ENOMEM ) ;
2008-07-18 20:00:18 +04:00
}
/* 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 ;
2010-09-17 14:31:41 +04:00
slave - > mtd . name = name ;
2008-07-18 20:00:18 +04:00
slave - > mtd . owner = master - > owner ;
2009-02-12 13:40:00 +03:00
slave - > mtd . backing_dev_info = master - > backing_dev_info ;
2008-07-18 20:00:18 +04:00
2009-03-26 10:42:41 +03:00
/* NOTE: we don't arrange MTDs as a tree; it'd be error-prone
* to have the same data be in two different partitions .
*/
slave - > mtd . dev . parent = master - > dev . parent ;
2008-07-18 20:00:18 +04:00
slave - > mtd . read = part_read ;
slave - > mtd . write = part_write ;
if ( master - > panic_write )
slave - > mtd . panic_write = part_panic_write ;
2008-07-18 20:00:33 +04:00
if ( master - > point & & master - > unpoint ) {
2008-07-18 20:00:18 +04:00
slave - > mtd . point = part_point ;
slave - > mtd . unpoint = part_unpoint ;
}
2009-02-12 13:40:00 +03:00
if ( master - > get_unmapped_area )
slave - > mtd . get_unmapped_area = part_get_unmapped_area ;
2008-07-18 20:00:18 +04:00
if ( master - > read_oob )
slave - > mtd . read_oob = part_read_oob ;
if ( master - > write_oob )
slave - > mtd . write_oob = part_write_oob ;
2008-07-18 20:00:33 +04:00
if ( master - > read_user_prot_reg )
2008-07-18 20:00:18 +04:00
slave - > mtd . read_user_prot_reg = part_read_user_prot_reg ;
2008-07-18 20:00:33 +04:00
if ( master - > read_fact_prot_reg )
2008-07-18 20:00:18 +04:00
slave - > mtd . read_fact_prot_reg = part_read_fact_prot_reg ;
2008-07-18 20:00:33 +04:00
if ( master - > write_user_prot_reg )
2008-07-18 20:00:18 +04:00
slave - > mtd . write_user_prot_reg = part_write_user_prot_reg ;
2008-07-18 20:00:33 +04:00
if ( master - > lock_user_prot_reg )
2008-07-18 20:00:18 +04:00
slave - > mtd . lock_user_prot_reg = part_lock_user_prot_reg ;
2008-07-18 20:00:33 +04:00
if ( master - > get_user_prot_info )
2008-07-18 20:00:18 +04:00
slave - > mtd . get_user_prot_info = part_get_user_prot_info ;
2008-07-18 20:00:33 +04:00
if ( master - > get_fact_prot_info )
2008-07-18 20:00:18 +04:00
slave - > mtd . get_fact_prot_info = part_get_fact_prot_info ;
if ( master - > sync )
slave - > mtd . sync = part_sync ;
2009-04-05 18:56:23 +04:00
if ( ! partno & & ! master - > dev . class & & master - > suspend & & master - > resume ) {
2008-07-18 20:00:18 +04:00
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 ;
2010-06-14 20:10:33 +04:00
if ( master - > is_locked )
slave - > mtd . is_locked = part_is_locked ;
2008-07-18 20:00:18 +04:00
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 ;
if ( slave - > offset = = MTDPART_OFS_APPEND )
slave - > offset = cur_offset ;
if ( slave - > offset = = MTDPART_OFS_NXTBLK ) {
slave - > offset = cur_offset ;
2008-12-10 16:37:21 +03:00
if ( mtd_mod_by_eb ( cur_offset , master ) ! = 0 ) {
2008-07-18 20:00:18 +04:00
/* Round up to next erasesize */
2008-12-10 16:37:21 +03:00
slave - > offset = ( mtd_div_by_eb ( cur_offset , master ) + 1 ) * master - > erasesize ;
2008-07-18 20:00:18 +04:00
printk ( KERN_NOTICE " Moving partition %d: "
2008-12-10 16:37:21 +03:00
" 0x%012llx -> 0x%012llx \n " , partno ,
( unsigned long long ) cur_offset , ( unsigned long long ) slave - > offset ) ;
2008-07-18 20:00:18 +04:00
}
}
if ( slave - > mtd . size = = MTDPART_SIZ_FULL )
slave - > mtd . size = master - > size - slave - > offset ;
2008-12-10 16:37:21 +03:00
printk ( KERN_NOTICE " 0x%012llx-0x%012llx : \" %s \" \n " , ( unsigned long long ) slave - > offset ,
( unsigned long long ) ( slave - > offset + slave - > mtd . size ) , slave - > mtd . name ) ;
2008-07-18 20:00:18 +04:00
/* let's do some sanity checks */
if ( slave - > offset > = master - > size ) {
2008-07-18 20:01:22 +04:00
/* let's register it anyway to preserve ordering */
2008-07-18 20:00:18 +04:00
slave - > offset = 0 ;
slave - > mtd . size = 0 ;
2008-07-18 20:00:33 +04:00
printk ( KERN_ERR " mtd: partition \" %s \" is out of reach -- disabled \n " ,
2008-07-18 20:00:18 +04:00
part - > name ) ;
2008-07-18 20:01:22 +04:00
goto out_register ;
2008-07-18 20:00:18 +04:00
}
if ( slave - > offset + slave - > mtd . size > master - > size ) {
slave - > mtd . size = master - > size - slave - > offset ;
2008-12-10 16:37:21 +03:00
printk ( KERN_WARNING " mtd: partition \" %s \" extends beyond the end of device \" %s \" -- size truncated to %#llx \n " ,
part - > name , master - > name , ( unsigned long long ) slave - > mtd . size ) ;
2008-07-18 20:00:18 +04:00
}
2008-07-18 20:00:33 +04:00
if ( master - > numeraseregions > 1 ) {
2008-07-18 20:00:18 +04:00
/* Deal with variable erase size stuff */
2008-07-18 20:00:57 +04:00
int i , max = master - > numeraseregions ;
2008-12-10 16:37:21 +03:00
u64 end = slave - > offset + slave - > mtd . size ;
2008-07-18 20:00:18 +04:00
struct mtd_erase_region_info * regions = master - > eraseregions ;
2008-07-18 20:00:57 +04:00
/* Find the first erase regions which is part of this
* partition . */
for ( i = 0 ; i < max & & regions [ i ] . offset < = slave - > offset ; i + + )
2008-07-18 20:00:18 +04:00
;
2008-07-18 20:00:57 +04:00
/* The loop searched for the region _behind_ the first one */
2009-09-18 23:51:50 +04:00
if ( i > 0 )
i - - ;
2008-07-18 20:00:18 +04:00
2008-07-18 20:00:57 +04:00
/* Pick biggest erasesize */
for ( ; i < max & & regions [ i ] . offset < end ; i + + ) {
2008-07-18 20:00:18 +04:00
if ( slave - > mtd . erasesize < regions [ i ] . erasesize ) {
slave - > mtd . erasesize = regions [ i ] . erasesize ;
}
}
2008-07-18 20:00:57 +04:00
BUG_ON ( slave - > mtd . erasesize = = 0 ) ;
2008-07-18 20:00:18 +04:00
} else {
/* Single erase size */
slave - > mtd . erasesize = master - > erasesize ;
}
if ( ( slave - > mtd . flags & MTD_WRITEABLE ) & &
2008-12-10 16:37:21 +03:00
mtd_mod_by_eb ( slave - > offset , & slave - > mtd ) ) {
2008-07-18 20:00:18 +04:00
/* Doesn't start on a boundary of major erase size */
2008-07-18 20:00:33 +04:00
/* FIXME: Let it be writable if it is on a boundary of
* _minor_ erase size though */
2008-07-18 20:00:18 +04:00
slave - > mtd . flags & = ~ MTD_WRITEABLE ;
2008-07-18 20:00:33 +04:00
printk ( KERN_WARNING " mtd: partition \" %s \" doesn't start on an erase block boundary -- force read-only \n " ,
2008-07-18 20:00:18 +04:00
part - > name ) ;
}
if ( ( slave - > mtd . flags & MTD_WRITEABLE ) & &
2008-12-10 16:37:21 +03:00
mtd_mod_by_eb ( slave - > mtd . size , & slave - > mtd ) ) {
2008-07-18 20:00:18 +04:00
slave - > mtd . flags & = ~ MTD_WRITEABLE ;
2008-07-18 20:00:33 +04:00
printk ( KERN_WARNING " mtd: partition \" %s \" doesn't end on an erase block -- force read-only \n " ,
2008-07-18 20:00:18 +04:00
part - > name ) ;
}
slave - > mtd . ecclayout = master - > ecclayout ;
if ( master - > block_isbad ) {
2008-12-10 16:37:21 +03:00
uint64_t offs = 0 ;
2008-07-18 20:00:18 +04:00
2008-07-18 20:00:33 +04:00
while ( offs < slave - > mtd . size ) {
2008-07-18 20:00:18 +04:00
if ( master - > block_isbad ( master ,
offs + slave - > offset ) )
slave - > mtd . ecc_stats . badblocks + + ;
offs + = slave - > mtd . erasesize ;
}
}
2008-07-18 20:01:22 +04:00
out_register :
2008-07-18 20:00:18 +04:00
return slave ;
}
2010-09-17 14:31:41 +04:00
int mtd_add_partition ( struct mtd_info * master , char * name ,
long long offset , long long length )
{
struct mtd_partition part ;
struct mtd_part * p , * new ;
uint64_t start , end ;
int ret = 0 ;
/* the direct offset is expected */
if ( offset = = MTDPART_OFS_APPEND | |
offset = = MTDPART_OFS_NXTBLK )
return - EINVAL ;
if ( length = = MTDPART_SIZ_FULL )
length = master - > size - offset ;
if ( length < = 0 )
return - EINVAL ;
part . name = name ;
part . size = length ;
part . offset = offset ;
part . mask_flags = 0 ;
part . ecclayout = NULL ;
new = allocate_partition ( master , & part , - 1 , offset ) ;
if ( IS_ERR ( new ) )
return PTR_ERR ( new ) ;
start = offset ;
end = offset + length ;
mutex_lock ( & mtd_partitions_mutex ) ;
list_for_each_entry ( p , & mtd_partitions , list )
if ( p - > master = = master ) {
if ( ( start > = p - > offset ) & &
( start < ( p - > offset + p - > mtd . size ) ) )
goto err_inv ;
if ( ( end > = p - > offset ) & &
( end < ( p - > offset + p - > mtd . size ) ) )
goto err_inv ;
}
list_add ( & new - > list , & mtd_partitions ) ;
mutex_unlock ( & mtd_partitions_mutex ) ;
add_mtd_device ( & new - > mtd ) ;
return ret ;
err_inv :
mutex_unlock ( & mtd_partitions_mutex ) ;
free_partition ( new ) ;
return - EINVAL ;
}
EXPORT_SYMBOL_GPL ( mtd_add_partition ) ;
int mtd_del_partition ( struct mtd_info * master , int partno )
{
struct mtd_part * slave , * next ;
int ret = - EINVAL ;
mutex_lock ( & mtd_partitions_mutex ) ;
list_for_each_entry_safe ( slave , next , & mtd_partitions , list )
if ( ( slave - > master = = master ) & &
( slave - > mtd . index = = partno ) ) {
ret = del_mtd_device ( & slave - > mtd ) ;
if ( ret < 0 )
break ;
list_del ( & slave - > list ) ;
free_partition ( slave ) ;
break ;
}
mutex_unlock ( & mtd_partitions_mutex ) ;
return ret ;
}
EXPORT_SYMBOL_GPL ( mtd_del_partition ) ;
2005-04-17 02:20:36 +04:00
/*
* 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 .
2009-03-26 10:42:41 +03:00
*
* We don ' t register the master , or expect the caller to have done so ,
* for reasons of data integrity .
2005-04-17 02:20:36 +04:00
*/
2005-11-07 14:15:26 +03:00
int add_mtd_partitions ( struct mtd_info * master ,
2005-04-17 02:20:36 +04:00
const struct mtd_partition * parts ,
int nbparts )
{
struct mtd_part * slave ;
2008-12-10 16:37:21 +03:00
uint64_t cur_offset = 0 ;
2005-04-17 02:20:36 +04:00
int i ;
2008-07-18 20:00:33 +04:00
printk ( KERN_NOTICE " Creating %d MTD partitions on \" %s \" : \n " , nbparts , master - > name ) ;
2005-04-17 02:20:36 +04:00
for ( i = 0 ; i < nbparts ; i + + ) {
2010-09-17 14:31:41 +04:00
slave = allocate_partition ( master , parts + i , i , cur_offset ) ;
if ( IS_ERR ( slave ) )
return PTR_ERR ( slave ) ;
mutex_lock ( & mtd_partitions_mutex ) ;
list_add ( & slave - > list , & mtd_partitions ) ;
mutex_unlock ( & mtd_partitions_mutex ) ;
add_mtd_device ( & slave - > mtd ) ;
2005-04-17 02:20:36 +04:00
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 )
{
2008-05-19 23:11:50 +04:00
struct mtd_part_parser * p , * ret = NULL ;
2005-04-17 02:20:36 +04:00
2008-05-19 23:11:50 +04:00
spin_lock ( & part_parser_lock ) ;
2005-04-17 02:20:36 +04:00
2008-05-19 23:11:50 +04:00
list_for_each_entry ( p , & part_parsers , list )
2005-04-17 02:20:36 +04:00
if ( ! strcmp ( p - > name , name ) & & try_module_get ( p - > owner ) ) {
ret = p ;
break ;
}
2008-05-19 23:11:50 +04:00
2005-04-17 02:20:36 +04:00
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 ;
}
2008-07-18 20:00:33 +04:00
EXPORT_SYMBOL_GPL ( register_mtd_parser ) ;
2005-04-17 02:20:36 +04:00
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 ;
}
2008-07-18 20:00:33 +04:00
EXPORT_SYMBOL_GPL ( deregister_mtd_parser ) ;
2005-04-17 02:20:36 +04:00
2005-11-07 14:15:26 +03:00
int parse_mtd_partitions ( struct mtd_info * master , const char * * types ,
2005-04-17 02:20:36 +04:00
struct mtd_partition * * pparts , unsigned long origin )
{
struct mtd_part_parser * parser ;
int ret = 0 ;
2005-11-07 14:15:26 +03:00
2005-04-17 02:20:36 +04:00
for ( ; ret < = 0 & & * types ; types + + ) {
parser = get_partition_parser ( * types ) ;
if ( ! parser & & ! request_module ( " %s " , * types ) )
parser = get_partition_parser ( * types ) ;
if ( ! parser ) {
printk ( KERN_NOTICE " %s partition parsing not available \n " ,
* types ) ;
continue ;
}
ret = ( * parser - > parse_fn ) ( master , pparts , origin ) ;
if ( ret > 0 ) {
2005-11-07 14:15:26 +03:00
printk ( KERN_NOTICE " %d %s partitions found on MTD device %s \n " ,
2005-04-17 02:20:36 +04:00
ret , parser - > name , master - > name ) ;
}
put_partition_parser ( parser ) ;
}
return ret ;
}
EXPORT_SYMBOL_GPL ( parse_mtd_partitions ) ;
2010-09-17 14:31:41 +04:00
int mtd_is_master ( struct mtd_info * mtd )
{
struct mtd_part * part ;
int nopart = 0 ;
mutex_lock ( & mtd_partitions_mutex ) ;
list_for_each_entry ( part , & mtd_partitions , list )
if ( & part - > mtd = = mtd ) {
nopart = 1 ;
break ;
}
mutex_unlock ( & mtd_partitions_mutex ) ;
return nopart ;
}
EXPORT_SYMBOL_GPL ( mtd_is_master ) ;