2007-07-20 21:39:26 +02:00
/*
* ( C ) Copyright IBM Deutschland Entwicklung GmbH 2006
*
* Author : Maxim Shchetynin < maxim @ de . ibm . com >
*
* Axon DDR2 device driver .
* It registers one block device per Axon ' s DDR2 memory bank found on a system .
* Block devices are called axonram ? , their major and minor numbers are
* available in / proc / devices , / proc / partitions or in / sys / block / axonram ? / dev .
*
* 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 , 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 . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include <linux/bio.h>
# include <linux/blkdev.h>
# include <linux/device.h>
# include <linux/errno.h>
# include <linux/fs.h>
# include <linux/genhd.h>
# include <linux/interrupt.h>
# include <linux/io.h>
# include <linux/ioport.h>
# include <linux/irq.h>
# include <linux/irqreturn.h>
# include <linux/kernel.h>
# include <linux/mm.h>
# include <linux/mod_devicetable.h>
# include <linux/module.h>
# include <linux/slab.h>
# include <linux/string.h>
# include <linux/types.h>
2007-11-14 04:13:09 +11:00
# include <linux/of_device.h>
# include <linux/of_platform.h>
2016-01-15 16:56:14 -08:00
# include <linux/pfn_t.h>
2007-11-14 04:13:09 +11:00
2007-07-20 21:39:26 +02:00
# include <asm/page.h>
# include <asm/prom.h>
# define AXON_RAM_MODULE_NAME "axonram"
# define AXON_RAM_DEVICE_NAME "axonram"
# define AXON_RAM_MINORS_PER_DISK 16
# define AXON_RAM_BLOCK_SHIFT PAGE_SHIFT
# define AXON_RAM_BLOCK_SIZE 1 << AXON_RAM_BLOCK_SHIFT
# define AXON_RAM_SECTOR_SHIFT 9
# define AXON_RAM_SECTOR_SIZE 1 << AXON_RAM_SECTOR_SHIFT
# define AXON_RAM_IRQ_FLAGS IRQF_SHARED | IRQF_TRIGGER_RISING
2008-07-16 05:51:41 +10:00
static int azfs_major , azfs_minor ;
2007-07-20 21:39:26 +02:00
struct axon_ram_bank {
2010-07-22 15:52:34 -06:00
struct platform_device * device ;
2007-07-20 21:39:26 +02:00
struct gendisk * disk ;
2007-08-23 03:01:27 +10:00
unsigned int irq_id ;
2007-07-20 21:39:26 +02:00
unsigned long ph_addr ;
unsigned long io_addr ;
unsigned long size ;
unsigned long ecc_counter ;
} ;
static ssize_t
axon_ram_sysfs_ecc ( struct device * dev , struct device_attribute * attr , char * buf )
{
2010-07-22 15:52:34 -06:00
struct platform_device * device = to_platform_device ( dev ) ;
2007-07-20 21:39:26 +02:00
struct axon_ram_bank * bank = device - > dev . platform_data ;
BUG_ON ( ! bank ) ;
return sprintf ( buf , " %ld \n " , bank - > ecc_counter ) ;
}
static DEVICE_ATTR ( ecc , S_IRUGO , axon_ram_sysfs_ecc , NULL ) ;
/**
* axon_ram_irq_handler - interrupt handler for Axon RAM ECC
* @ irq : interrupt ID
* @ dev : pointer to of_device
*/
static irqreturn_t
axon_ram_irq_handler ( int irq , void * dev )
{
2010-07-22 15:52:34 -06:00
struct platform_device * device = dev ;
2007-07-20 21:39:26 +02:00
struct axon_ram_bank * bank = device - > dev . platform_data ;
BUG_ON ( ! bank ) ;
2011-03-30 22:57:33 -03:00
dev_err ( & device - > dev , " Correctable memory error occurred \n " ) ;
2007-08-23 03:01:27 +10:00
bank - > ecc_counter + + ;
return IRQ_HANDLED ;
2007-07-20 21:39:26 +02:00
}
/**
* axon_ram_make_request - make_request ( ) method for block device
* @ queue , @ bio : see blk_queue_make_request ( )
*/
2015-11-05 10:41:16 -07:00
static blk_qc_t
2007-07-20 21:39:26 +02:00
axon_ram_make_request ( struct request_queue * queue , struct bio * bio )
{
struct axon_ram_bank * bank = bio - > bi_bdev - > bd_disk - > private_data ;
unsigned long phys_mem , phys_end ;
void * user_mem ;
2013-11-23 17:19:00 -08:00
struct bio_vec vec ;
2007-07-20 21:39:26 +02:00
unsigned int transfered ;
2013-11-23 17:19:00 -08:00
struct bvec_iter iter ;
2007-07-20 21:39:26 +02:00
2013-10-11 15:44:27 -07:00
phys_mem = bank - > io_addr + ( bio - > bi_iter . bi_sector < <
AXON_RAM_SECTOR_SHIFT ) ;
2007-07-20 21:39:26 +02:00
phys_end = bank - > io_addr + bank - > size ;
transfered = 0 ;
2013-11-23 17:19:00 -08:00
bio_for_each_segment ( vec , bio , iter ) {
if ( unlikely ( phys_mem + vec . bv_len > phys_end ) ) {
2007-10-12 07:00:19 +01:00
bio_io_error ( bio ) ;
2015-11-05 10:41:16 -07:00
return BLK_QC_T_NONE ;
2007-07-20 21:39:26 +02:00
}
2013-11-23 17:19:00 -08:00
user_mem = page_address ( vec . bv_page ) + vec . bv_offset ;
2007-07-20 21:39:26 +02:00
if ( bio_data_dir ( bio ) = = READ )
2013-11-23 17:19:00 -08:00
memcpy ( user_mem , ( void * ) phys_mem , vec . bv_len ) ;
2007-07-20 21:39:26 +02:00
else
2013-11-23 17:19:00 -08:00
memcpy ( ( void * ) phys_mem , user_mem , vec . bv_len ) ;
2007-07-20 21:39:26 +02:00
2013-11-23 17:19:00 -08:00
phys_mem + = vec . bv_len ;
transfered + = vec . bv_len ;
2007-07-20 21:39:26 +02:00
}
2015-07-20 15:29:37 +02:00
bio_endio ( bio ) ;
2015-11-05 10:41:16 -07:00
return BLK_QC_T_NONE ;
2007-07-20 21:39:26 +02:00
}
/**
* axon_ram_direct_access - direct_access ( ) method for block device
* @ device , @ sector , @ data : see block_device_operations method
*/
2015-01-07 18:05:34 +02:00
static long
2007-07-20 21:39:26 +02:00
axon_ram_direct_access ( struct block_device * device , sector_t sector ,
2016-01-15 16:56:14 -08:00
void __pmem * * kaddr , pfn_t * pfn )
2007-07-20 21:39:26 +02:00
{
struct axon_ram_bank * bank = device - > bd_disk - > private_data ;
2015-01-07 18:05:34 +02:00
loff_t offset = ( loff_t ) sector < < AXON_RAM_SECTOR_SHIFT ;
2007-07-20 21:39:26 +02:00
2016-01-15 16:56:14 -08:00
* kaddr = ( void __pmem __force * ) bank - > io_addr + offset ;
* pfn = phys_to_pfn_t ( bank - > ph_addr + offset , PFN_DEV ) ;
2015-01-07 18:05:34 +02:00
return bank - > size - offset ;
2007-07-20 21:39:26 +02:00
}
2009-09-21 17:01:13 -07:00
static const struct block_device_operations axon_ram_devops = {
2007-07-20 21:39:26 +02:00
. owner = THIS_MODULE ,
. direct_access = axon_ram_direct_access
} ;
/**
* axon_ram_probe - probe ( ) method for platform driver
2011-02-22 19:59:54 -07:00
* @ device : see platform_driver method
2007-07-20 21:39:26 +02:00
*/
2011-02-22 19:59:54 -07:00
static int axon_ram_probe ( struct platform_device * device )
2007-07-20 21:39:26 +02:00
{
static int axon_ram_bank_id = - 1 ;
struct axon_ram_bank * bank ;
struct resource resource ;
int rc = 0 ;
axon_ram_bank_id + + ;
dev_info ( & device - > dev , " Found memory controller on %s \n " ,
2010-04-13 16:12:29 -07:00
device - > dev . of_node - > full_name ) ;
2007-07-20 21:39:26 +02:00
bank = kzalloc ( sizeof ( struct axon_ram_bank ) , GFP_KERNEL ) ;
if ( bank = = NULL ) {
dev_err ( & device - > dev , " Out of memory \n " ) ;
rc = - ENOMEM ;
goto failed ;
}
device - > dev . platform_data = bank ;
bank - > device = device ;
2010-04-13 16:12:29 -07:00
if ( of_address_to_resource ( device - > dev . of_node , 0 , & resource ) ! = 0 ) {
2007-07-20 21:39:26 +02:00
dev_err ( & device - > dev , " Cannot access device tree \n " ) ;
rc = - EFAULT ;
goto failed ;
}
2011-06-09 09:13:32 -07:00
bank - > size = resource_size ( & resource ) ;
2007-07-20 21:39:26 +02:00
if ( bank - > size = = 0 ) {
dev_err ( & device - > dev , " No DDR2 memory found for %s%d \n " ,
AXON_RAM_DEVICE_NAME , axon_ram_bank_id ) ;
rc = - ENODEV ;
goto failed ;
}
dev_info ( & device - > dev , " Register DDR2 memory device %s%d with %luMB \n " ,
AXON_RAM_DEVICE_NAME , axon_ram_bank_id , bank - > size > > 20 ) ;
bank - > ph_addr = resource . start ;
2011-05-08 21:43:47 +00:00
bank - > io_addr = ( unsigned long ) ioremap_prot (
2007-07-20 21:39:26 +02:00
bank - > ph_addr , bank - > size , _PAGE_NO_CACHE ) ;
if ( bank - > io_addr = = 0 ) {
dev_err ( & device - > dev , " ioremap() failed \n " ) ;
rc = - EFAULT ;
goto failed ;
}
bank - > disk = alloc_disk ( AXON_RAM_MINORS_PER_DISK ) ;
if ( bank - > disk = = NULL ) {
dev_err ( & device - > dev , " Cannot register disk \n " ) ;
rc = - EFAULT ;
goto failed ;
}
2008-07-16 05:51:41 +10:00
bank - > disk - > major = azfs_major ;
bank - > disk - > first_minor = azfs_minor ;
2007-07-20 21:39:26 +02:00
bank - > disk - > fops = & axon_ram_devops ;
bank - > disk - > private_data = bank ;
bank - > disk - > driverfs_dev = & device - > dev ;
sprintf ( bank - > disk - > disk_name , " %s%d " ,
AXON_RAM_DEVICE_NAME , axon_ram_bank_id ) ;
bank - > disk - > queue = blk_alloc_queue ( GFP_KERNEL ) ;
if ( bank - > disk - > queue = = NULL ) {
dev_err ( & device - > dev , " Cannot register disk queue \n " ) ;
rc = - EFAULT ;
goto failed ;
}
set_capacity ( bank - > disk , bank - > size > > AXON_RAM_SECTOR_SHIFT ) ;
blk_queue_make_request ( bank - > disk - > queue , axon_ram_make_request ) ;
2009-05-22 17:17:49 -04:00
blk_queue_logical_block_size ( bank - > disk - > queue , AXON_RAM_SECTOR_SIZE ) ;
2007-07-20 21:39:26 +02:00
add_disk ( bank - > disk ) ;
2010-04-13 16:12:29 -07:00
bank - > irq_id = irq_of_parse_and_map ( device - > dev . of_node , 0 ) ;
2007-08-23 03:01:27 +10:00
if ( bank - > irq_id = = NO_IRQ ) {
2007-07-20 21:39:26 +02:00
dev_err ( & device - > dev , " Cannot access ECC interrupt ID \n " ) ;
rc = - EFAULT ;
goto failed ;
}
2007-08-23 03:01:27 +10:00
rc = request_irq ( bank - > irq_id , axon_ram_irq_handler ,
2007-07-20 21:39:26 +02:00
AXON_RAM_IRQ_FLAGS , bank - > disk - > disk_name , device ) ;
if ( rc ! = 0 ) {
dev_err ( & device - > dev , " Cannot register ECC interrupt handler \n " ) ;
2007-08-23 03:01:27 +10:00
bank - > irq_id = NO_IRQ ;
2007-07-20 21:39:26 +02:00
rc = - EFAULT ;
goto failed ;
}
rc = device_create_file ( & device - > dev , & dev_attr_ecc ) ;
if ( rc ! = 0 ) {
dev_err ( & device - > dev , " Cannot create sysfs file \n " ) ;
rc = - EFAULT ;
goto failed ;
}
2008-07-16 05:51:41 +10:00
azfs_minor + = bank - > disk - > minors ;
2007-07-20 21:39:26 +02:00
return 0 ;
failed :
if ( bank ! = NULL ) {
2007-08-23 03:01:27 +10:00
if ( bank - > irq_id ! = NO_IRQ )
free_irq ( bank - > irq_id , device ) ;
2007-07-20 21:39:26 +02:00
if ( bank - > disk ! = NULL ) {
if ( bank - > disk - > major > 0 )
unregister_blkdev ( bank - > disk - > major ,
bank - > disk - > disk_name ) ;
del_gendisk ( bank - > disk ) ;
}
device - > dev . platform_data = NULL ;
if ( bank - > io_addr ! = 0 )
iounmap ( ( void __iomem * ) bank - > io_addr ) ;
kfree ( bank ) ;
}
return rc ;
}
/**
* axon_ram_remove - remove ( ) method for platform driver
* @ device : see of_platform_driver method
*/
static int
2010-07-22 15:52:34 -06:00
axon_ram_remove ( struct platform_device * device )
2007-07-20 21:39:26 +02:00
{
struct axon_ram_bank * bank = device - > dev . platform_data ;
BUG_ON ( ! bank | | ! bank - > disk ) ;
device_remove_file ( & device - > dev , & dev_attr_ecc ) ;
2007-08-23 03:01:27 +10:00
free_irq ( bank - > irq_id , device ) ;
2007-07-20 21:39:26 +02:00
del_gendisk ( bank - > disk ) ;
iounmap ( ( void __iomem * ) bank - > io_addr ) ;
kfree ( bank ) ;
return 0 ;
}
2014-09-10 21:56:38 +02:00
static const struct of_device_id axon_ram_device_id [ ] = {
2007-07-20 21:39:26 +02:00
{
. type = " dma-memory "
} ,
{ }
} ;
2015-10-20 16:04:13 +01:00
MODULE_DEVICE_TABLE ( of , axon_ram_device_id ) ;
2007-07-20 21:39:26 +02:00
2011-02-22 19:59:54 -07:00
static struct platform_driver axon_ram_driver = {
2007-07-20 21:39:26 +02:00
. probe = axon_ram_probe ,
2007-10-11 15:19:03 +10:00
. remove = axon_ram_remove ,
2010-04-13 16:13:02 -07:00
. driver = {
. name = AXON_RAM_MODULE_NAME ,
. of_match_table = axon_ram_device_id ,
2007-10-11 15:19:03 +10:00
} ,
2007-07-20 21:39:26 +02:00
} ;
/**
* axon_ram_init
*/
static int __init
axon_ram_init ( void )
{
2008-07-16 05:51:41 +10:00
azfs_major = register_blkdev ( azfs_major , AXON_RAM_DEVICE_NAME ) ;
if ( azfs_major < 0 ) {
printk ( KERN_ERR " %s cannot become block device major number \n " ,
AXON_RAM_MODULE_NAME ) ;
return - EFAULT ;
}
azfs_minor = 0 ;
2011-02-22 19:59:54 -07:00
return platform_driver_register ( & axon_ram_driver ) ;
2007-07-20 21:39:26 +02:00
}
/**
* axon_ram_exit
*/
static void __exit
axon_ram_exit ( void )
{
2011-02-22 19:59:54 -07:00
platform_driver_unregister ( & axon_ram_driver ) ;
2008-07-16 05:51:41 +10:00
unregister_blkdev ( azfs_major , AXON_RAM_DEVICE_NAME ) ;
2007-07-20 21:39:26 +02:00
}
module_init ( axon_ram_init ) ;
module_exit ( axon_ram_exit ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_AUTHOR ( " Maxim Shchetynin <maxim@de.ibm.com> " ) ;
MODULE_DESCRIPTION ( " Axon DDR2 RAM device driver for IBM Cell BE " ) ;