2005-04-16 15:20:36 -07:00
/*
* mtdram - a test mtd device
* Author : Alexander Larsson < alex @ cendio . se >
*
* Copyright ( c ) 1999 Alexander Larsson < alex @ cendio . se >
2005-04-21 04:42:15 +01:00
* Copyright ( c ) 2005 Joern Engel < joern @ wh . fh - wedel . de >
2005-04-16 15:20:36 -07:00
*
* This code is GPL
*
*/
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/ioport.h>
# include <linux/vmalloc.h>
2017-10-30 14:48:29 -04:00
# include <linux/mm.h>
2005-04-16 15:20:36 -07:00
# include <linux/init.h>
# include <linux/mtd/mtd.h>
2008-04-14 17:20:02 +03:00
# include <linux/mtd/mtdram.h>
2005-04-16 15:20:36 -07:00
static unsigned long total_size = CONFIG_MTDRAM_TOTAL_SIZE ;
static unsigned long erase_size = CONFIG_MTDRAM_ERASE_SIZE ;
2016-03-02 16:26:59 +01:00
static unsigned long writebuf_size = 64 ;
2005-04-16 15:20:36 -07:00
# define MTDRAM_TOTAL_SIZE (total_size * 1024)
# define MTDRAM_ERASE_SIZE (erase_size * 1024)
2005-04-21 04:42:15 +01: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 16:26:59 +01:00
module_param ( writebuf_size , ulong , 0 ) ;
MODULE_PARM_DESC ( writebuf_size , " Device write buf size in Bytes (Default: 64) " ) ;
2005-04-21 04:42:15 +01:00
# endif
2005-04-16 15:20:36 -07:00
// We could store these in the mtd structure, but we only support 1 device..
static struct mtd_info * mtd_info ;
2015-09-30 09:01:19 +08: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 04:42:15 +01:00
static int ram_erase ( struct mtd_info * mtd , struct erase_info * instr )
2005-04-16 15:20:36 -07:00
{
2015-09-30 09:01:19 +08:00
if ( check_offs_len ( mtd , instr - > addr , instr - > len ) )
return - EINVAL ;
2005-04-21 04:42:15 +01:00
memset ( ( char * ) mtd - > priv + instr - > addr , 0xff , instr - > len ) ;
2018-02-12 22:03:11 +01:00
2005-04-21 04:42:15 +01:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2005-04-21 04:42:15 +01:00
static int ram_point ( struct mtd_info * mtd , loff_t from , size_t len ,
2008-04-29 23:26:49 -07:00
size_t * retlen , void * * virt , resource_size_t * phys )
2005-04-16 15:20:36 -07:00
{
2008-04-29 23:26:49 -07:00
* virt = mtd - > priv + from ;
2005-04-21 04:42:15 +01:00
* retlen = len ;
2017-10-30 14:48:29 -04:00
if ( phys ) {
/* limit retlen to the number of contiguous physical pages */
unsigned long page_ofs = offset_in_page ( * virt ) ;
void * addr = * virt - page_ofs ;
unsigned long pfn1 , pfn0 = vmalloc_to_pfn ( addr ) ;
* phys = __pfn_to_phys ( pfn0 ) + page_ofs ;
len + = page_ofs ;
while ( len > PAGE_SIZE ) {
len - = PAGE_SIZE ;
addr + = PAGE_SIZE ;
pfn0 + + ;
pfn1 = vmalloc_to_pfn ( addr ) ;
if ( pfn1 ! = pfn0 ) {
* retlen = addr - * virt ;
break ;
}
}
}
2005-04-21 04:42:15 +01:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2012-02-03 13:20:43 +02:00
static int ram_unpoint ( struct mtd_info * mtd , loff_t from , size_t len )
2005-04-16 15:20:36 -07:00
{
2012-02-03 13:20:43 +02:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
static int ram_read ( struct mtd_info * mtd , loff_t from , size_t len ,
2005-04-21 04:42:15 +01:00
size_t * retlen , u_char * buf )
2005-04-16 15:20:36 -07:00
{
2005-04-21 04:42:15 +01:00
memcpy ( buf , mtd - > priv + from , len ) ;
* retlen = len ;
return 0 ;
2005-04-16 15:20:36 -07:00
}
static int ram_write ( struct mtd_info * mtd , loff_t to , size_t len ,
2005-04-21 04:42:15 +01:00
size_t * retlen , const u_char * buf )
2005-04-16 15:20:36 -07:00
{
2005-04-21 04:42:15 +01:00
memcpy ( ( char * ) mtd - > priv + to , buf , len ) ;
* retlen = len ;
return 0 ;
2005-04-16 15:20:36 -07:00
}
static void __exit cleanup_mtdram ( void )
{
2005-04-21 04:42:15 +01:00
if ( mtd_info ) {
2011-05-23 10:23:40 +01:00
mtd_device_unregister ( mtd_info ) ;
2005-09-10 00:26:54 -07:00
vfree ( mtd_info - > priv ) ;
2005-04-21 04:42:15 +01:00
kfree ( mtd_info ) ;
}
2005-04-16 15:20:36 -07:00
}
2005-04-21 04:42:15 +01:00
int mtdram_init_device ( struct mtd_info * mtd , void * mapped_address ,
2014-01-11 15:51:45 -08:00
unsigned long size , const char * name )
2005-04-16 15:20:36 -07:00
{
2005-04-21 04:42:15 +01:00
memset ( mtd , 0 , sizeof ( * mtd ) ) ;
/* Setup the MTD structure */
mtd - > name = name ;
2006-06-14 21:39:48 +01:00
mtd - > type = MTD_RAM ;
2005-04-21 04:42:15 +01:00
mtd - > flags = MTD_CAP_RAM ;
mtd - > size = size ;
2006-06-14 19:53:44 +04:00
mtd - > writesize = 1 ;
2016-03-02 16:26:59 +01:00
mtd - > writebufsize = writebuf_size ;
2005-04-21 04:42:15 +01:00
mtd - > erasesize = MTDRAM_ERASE_SIZE ;
mtd - > priv = mapped_address ;
mtd - > owner = THIS_MODULE ;
2012-01-30 14:58:32 +02:00
mtd - > _erase = ram_erase ;
mtd - > _point = ram_point ;
mtd - > _unpoint = ram_unpoint ;
mtd - > _read = ram_read ;
mtd - > _write = ram_write ;
2005-04-21 04:42:15 +01:00
2011-05-23 10:23:40 +01:00
if ( mtd_device_register ( mtd , NULL , 0 ) )
2005-04-21 04:42:15 +01:00
return - EIO ;
return 0 ;
2005-04-16 15:20:36 -07:00
}
static int __init init_mtdram ( void )
{
2005-04-21 04:42:15 +01: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-16 15:20:36 -07: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 " ) ;