2005-04-17 02:20:36 +04:00
/*
* mtdram - a test mtd device
* Author : Alexander Larsson < alex @ cendio . se >
*
* Copyright ( c ) 1999 Alexander Larsson < alex @ cendio . se >
2005-04-21 07:42:15 +04:00
* Copyright ( c ) 2005 Joern Engel < joern @ wh . fh - wedel . de >
2005-04-17 02:20:36 +04:00
*
* This code is GPL
*
*/
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/ioport.h>
# include <linux/vmalloc.h>
# include <linux/init.h>
# include <linux/mtd/mtd.h>
2008-04-14 18:20:02 +04:00
# include <linux/mtd/mtdram.h>
2005-04-17 02:20:36 +04:00
static unsigned long total_size = CONFIG_MTDRAM_TOTAL_SIZE ;
static unsigned long erase_size = CONFIG_MTDRAM_ERASE_SIZE ;
2016-03-02 18:26:59 +03:00
static unsigned long writebuf_size = 64 ;
2005-04-17 02:20:36 +04:00
# define MTDRAM_TOTAL_SIZE (total_size * 1024)
# define MTDRAM_ERASE_SIZE (erase_size * 1024)
2005-04-21 07:42:15 +04:00
# ifdef MODULE
module_param ( total_size , ulong , 0 ) ;
MODULE_PARM_DESC ( total_size , " Total device size in KiB " ) ;
module_param ( erase_size , ulong , 0 ) ;
MODULE_PARM_DESC ( erase_size , " Device erase block size in KiB " ) ;
2016-03-02 18:26:59 +03:00
module_param ( writebuf_size , ulong , 0 ) ;
MODULE_PARM_DESC ( writebuf_size , " Device write buf size in Bytes (Default: 64) " ) ;
2005-04-21 07:42:15 +04:00
# endif
2005-04-17 02:20:36 +04:00
// We could store these in the mtd structure, but we only support 1 device..
static struct mtd_info * mtd_info ;
2015-09-30 04:01:19 +03:00
static int check_offs_len ( struct mtd_info * mtd , loff_t ofs , uint64_t len )
{
int ret = 0 ;
/* Start address must align on block boundary */
if ( mtd_mod_by_eb ( ofs , mtd ) ) {
pr_debug ( " %s: unaligned address \n " , __func__ ) ;
ret = - EINVAL ;
}
/* Length must align on block boundary */
if ( mtd_mod_by_eb ( len , mtd ) ) {
pr_debug ( " %s: length not block aligned \n " , __func__ ) ;
ret = - EINVAL ;
}
return ret ;
}
2005-04-21 07:42:15 +04:00
static int ram_erase ( struct mtd_info * mtd , struct erase_info * instr )
2005-04-17 02:20:36 +04:00
{
2015-09-30 04:01:19 +03:00
if ( check_offs_len ( mtd , instr - > addr , instr - > len ) )
return - EINVAL ;
2005-04-21 07:42:15 +04:00
memset ( ( char * ) mtd - > priv + instr - > addr , 0xff , instr - > len ) ;
instr - > state = MTD_ERASE_DONE ;
mtd_erase_callback ( instr ) ;
return 0 ;
2005-04-17 02:20:36 +04:00
}
2005-04-21 07:42:15 +04:00
static int ram_point ( struct mtd_info * mtd , loff_t from , size_t len ,
2008-04-30 10:26:49 +04:00
size_t * retlen , void * * virt , resource_size_t * phys )
2005-04-17 02:20:36 +04:00
{
2008-04-30 10:26:49 +04:00
* virt = mtd - > priv + from ;
2005-04-21 07:42:15 +04:00
* retlen = len ;
return 0 ;
2005-04-17 02:20:36 +04:00
}
2012-02-03 15:20:43 +04:00
static int ram_unpoint ( struct mtd_info * mtd , loff_t from , size_t len )
2005-04-17 02:20:36 +04:00
{
2012-02-03 15:20:43 +04:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
2009-02-12 13:40:00 +03:00
/*
* Allow NOMMU mmap ( ) to directly map the device ( if not NULL )
* - return the address to which the offset maps
* - return - ENOSYS to indicate refusal to do the mapping
*/
static unsigned long ram_get_unmapped_area ( struct mtd_info * mtd ,
unsigned long len ,
unsigned long offset ,
unsigned long flags )
{
return ( unsigned long ) mtd - > priv + offset ;
}
2005-04-17 02:20:36 +04:00
static int ram_read ( struct mtd_info * mtd , loff_t from , size_t len ,
2005-04-21 07:42:15 +04:00
size_t * retlen , u_char * buf )
2005-04-17 02:20:36 +04:00
{
2005-04-21 07:42:15 +04:00
memcpy ( buf , mtd - > priv + from , len ) ;
* retlen = len ;
return 0 ;
2005-04-17 02:20:36 +04:00
}
static int ram_write ( struct mtd_info * mtd , loff_t to , size_t len ,
2005-04-21 07:42:15 +04:00
size_t * retlen , const u_char * buf )
2005-04-17 02:20:36 +04:00
{
2005-04-21 07:42:15 +04:00
memcpy ( ( char * ) mtd - > priv + to , buf , len ) ;
* retlen = len ;
return 0 ;
2005-04-17 02:20:36 +04:00
}
static void __exit cleanup_mtdram ( void )
{
2005-04-21 07:42:15 +04:00
if ( mtd_info ) {
2011-05-23 13:23:40 +04:00
mtd_device_unregister ( mtd_info ) ;
2005-09-10 11:26:54 +04:00
vfree ( mtd_info - > priv ) ;
2005-04-21 07:42:15 +04:00
kfree ( mtd_info ) ;
}
2005-04-17 02:20:36 +04:00
}
2005-04-21 07:42:15 +04:00
int mtdram_init_device ( struct mtd_info * mtd , void * mapped_address ,
2014-01-12 03:51:45 +04:00
unsigned long size , const char * name )
2005-04-17 02:20:36 +04:00
{
2005-04-21 07:42:15 +04:00
memset ( mtd , 0 , sizeof ( * mtd ) ) ;
/* Setup the MTD structure */
mtd - > name = name ;
2006-06-15 00:39:48 +04:00
mtd - > type = MTD_RAM ;
2005-04-21 07:42:15 +04:00
mtd - > flags = MTD_CAP_RAM ;
mtd - > size = size ;
2006-06-14 19:53:44 +04:00
mtd - > writesize = 1 ;
2016-03-02 18:26:59 +03:00
mtd - > writebufsize = writebuf_size ;
2005-04-21 07:42:15 +04:00
mtd - > erasesize = MTDRAM_ERASE_SIZE ;
mtd - > priv = mapped_address ;
mtd - > owner = THIS_MODULE ;
2012-01-30 16:58:32 +04:00
mtd - > _erase = ram_erase ;
mtd - > _point = ram_point ;
mtd - > _unpoint = ram_unpoint ;
mtd - > _get_unmapped_area = ram_get_unmapped_area ;
mtd - > _read = ram_read ;
mtd - > _write = ram_write ;
2005-04-21 07:42:15 +04:00
2011-05-23 13:23:40 +04:00
if ( mtd_device_register ( mtd , NULL , 0 ) )
2005-04-21 07:42:15 +04:00
return - EIO ;
return 0 ;
2005-04-17 02:20:36 +04:00
}
static int __init init_mtdram ( void )
{
2005-04-21 07:42:15 +04:00
void * addr ;
int err ;
if ( ! total_size )
return - EINVAL ;
/* Allocate some memory */
mtd_info = kmalloc ( sizeof ( struct mtd_info ) , GFP_KERNEL ) ;
if ( ! mtd_info )
return - ENOMEM ;
addr = vmalloc ( MTDRAM_TOTAL_SIZE ) ;
if ( ! addr ) {
kfree ( mtd_info ) ;
mtd_info = NULL ;
return - ENOMEM ;
}
err = mtdram_init_device ( mtd_info , addr , MTDRAM_TOTAL_SIZE , " mtdram test device " ) ;
if ( err ) {
vfree ( addr ) ;
kfree ( mtd_info ) ;
mtd_info = NULL ;
return err ;
}
memset ( mtd_info - > priv , 0xff , MTDRAM_TOTAL_SIZE ) ;
return err ;
2005-04-17 02:20:36 +04:00
}
module_init ( init_mtdram ) ;
module_exit ( cleanup_mtdram ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " Alexander Larsson <alexl@redhat.com> " ) ;
MODULE_DESCRIPTION ( " Simulated MTD driver for testing " ) ;