2012-06-13 21:01:55 +04:00
/*
* Copyright 2011 - 2012 Calxeda , Inc .
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms and conditions of the GNU General Public License ,
* version 2 , as published by the Free Software Foundation .
*
* This program is distributed in the hope 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 , see < http : //www.gnu.org/licenses/>.
*/
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/ctype.h>
# include <linux/edac.h>
# include <linux/interrupt.h>
# include <linux/platform_device.h>
# include <linux/of_platform.h>
# include <linux/uaccess.h>
# include "edac_core.h"
# include "edac_module.h"
/* DDR Ctrlr Error Registers */
2013-10-10 19:32:30 +04:00
# define HB_DDR_ECC_ERR_BASE 0x128
# define MW_DDR_ECC_ERR_BASE 0x1b4
# define HB_DDR_ECC_OPT 0x00
# define HB_DDR_ECC_U_ERR_ADDR 0x08
# define HB_DDR_ECC_U_ERR_STAT 0x0c
# define HB_DDR_ECC_U_ERR_DATAL 0x10
# define HB_DDR_ECC_U_ERR_DATAH 0x14
# define HB_DDR_ECC_C_ERR_ADDR 0x18
# define HB_DDR_ECC_C_ERR_STAT 0x1c
# define HB_DDR_ECC_C_ERR_DATAL 0x20
# define HB_DDR_ECC_C_ERR_DATAH 0x24
2012-06-13 21:01:55 +04:00
# define HB_DDR_ECC_OPT_MODE_MASK 0x3
# define HB_DDR_ECC_OPT_FWC 0x100
# define HB_DDR_ECC_OPT_XOR_SHIFT 16
2013-10-10 19:32:30 +04:00
/* DDR Ctrlr Interrupt Registers */
# define HB_DDR_ECC_INT_BASE 0x180
# define MW_DDR_ECC_INT_BASE 0x218
# define HB_DDR_ECC_INT_STATUS 0x00
# define HB_DDR_ECC_INT_ACK 0x04
# define HB_DDR_ECC_INT_STAT_CE 0x8
# define HB_DDR_ECC_INT_STAT_DOUBLE_CE 0x10
# define HB_DDR_ECC_INT_STAT_UE 0x20
# define HB_DDR_ECC_INT_STAT_DOUBLE_UE 0x40
2012-06-13 21:01:55 +04:00
struct hb_mc_drvdata {
2013-10-10 19:32:30 +04:00
void __iomem * mc_err_base ;
void __iomem * mc_int_base ;
2012-06-13 21:01:55 +04:00
} ;
static irqreturn_t highbank_mc_err_handler ( int irq , void * dev_id )
{
struct mem_ctl_info * mci = dev_id ;
struct hb_mc_drvdata * drvdata = mci - > pvt_info ;
u32 status , err_addr ;
/* Read the interrupt status register */
2013-10-10 19:32:30 +04:00
status = readl ( drvdata - > mc_int_base + HB_DDR_ECC_INT_STATUS ) ;
2012-06-13 21:01:55 +04:00
if ( status & HB_DDR_ECC_INT_STAT_UE ) {
2013-10-10 19:32:30 +04:00
err_addr = readl ( drvdata - > mc_err_base + HB_DDR_ECC_U_ERR_ADDR ) ;
2012-06-13 21:01:55 +04:00
edac_mc_handle_error ( HW_EVENT_ERR_UNCORRECTED , mci , 1 ,
err_addr > > PAGE_SHIFT ,
err_addr & ~ PAGE_MASK , 0 ,
0 , 0 , - 1 ,
mci - > ctl_name , " " ) ;
}
if ( status & HB_DDR_ECC_INT_STAT_CE ) {
2013-10-10 19:32:30 +04:00
u32 syndrome = readl ( drvdata - > mc_err_base + HB_DDR_ECC_C_ERR_STAT ) ;
2012-06-13 21:01:55 +04:00
syndrome = ( syndrome > > 8 ) & 0xff ;
2013-10-10 19:32:30 +04:00
err_addr = readl ( drvdata - > mc_err_base + HB_DDR_ECC_C_ERR_ADDR ) ;
2012-06-13 21:01:55 +04:00
edac_mc_handle_error ( HW_EVENT_ERR_CORRECTED , mci , 1 ,
err_addr > > PAGE_SHIFT ,
err_addr & ~ PAGE_MASK , syndrome ,
0 , 0 , - 1 ,
mci - > ctl_name , " " ) ;
}
/* clear the error, clears the interrupt */
2013-10-10 19:32:30 +04:00
writel ( status , drvdata - > mc_int_base + HB_DDR_ECC_INT_ACK ) ;
2012-06-13 21:01:55 +04:00
return IRQ_HANDLED ;
}
# ifdef CONFIG_EDAC_DEBUG
static ssize_t highbank_mc_err_inject_write ( struct file * file ,
const char __user * data ,
size_t count , loff_t * ppos )
{
struct mem_ctl_info * mci = file - > private_data ;
struct hb_mc_drvdata * pdata = mci - > pvt_info ;
char buf [ 32 ] ;
size_t buf_size ;
u32 reg ;
u8 synd ;
buf_size = min ( count , ( sizeof ( buf ) - 1 ) ) ;
if ( copy_from_user ( buf , data , buf_size ) )
return - EFAULT ;
buf [ buf_size ] = 0 ;
if ( ! kstrtou8 ( buf , 16 , & synd ) ) {
2013-10-10 19:32:30 +04:00
reg = readl ( pdata - > mc_err_base + HB_DDR_ECC_OPT ) ;
2012-06-13 21:01:55 +04:00
reg & = HB_DDR_ECC_OPT_MODE_MASK ;
reg | = ( synd < < HB_DDR_ECC_OPT_XOR_SHIFT ) | HB_DDR_ECC_OPT_FWC ;
2013-10-10 19:32:30 +04:00
writel ( reg , pdata - > mc_err_base + HB_DDR_ECC_OPT ) ;
2012-06-13 21:01:55 +04:00
}
return count ;
}
static const struct file_operations highbank_mc_debug_inject_fops = {
2012-10-17 12:30:27 +04:00
. open = simple_open ,
2012-06-13 21:01:55 +04:00
. write = highbank_mc_err_inject_write ,
. llseek = generic_file_llseek ,
} ;
2012-12-22 01:23:51 +04:00
static void highbank_mc_create_debugfs_nodes ( struct mem_ctl_info * mci )
2012-06-13 21:01:55 +04:00
{
if ( mci - > debugfs )
debugfs_create_file ( " inject_ctrl " , S_IWUSR , mci - > debugfs , mci ,
& highbank_mc_debug_inject_fops ) ;
;
}
# else
2012-12-22 01:23:51 +04:00
static void highbank_mc_create_debugfs_nodes ( struct mem_ctl_info * mci )
2012-06-13 21:01:55 +04:00
{ }
# endif
2013-10-10 19:32:30 +04:00
struct hb_mc_settings {
int err_offset ;
int int_offset ;
} ;
static struct hb_mc_settings hb_settings = {
. err_offset = HB_DDR_ECC_ERR_BASE ,
. int_offset = HB_DDR_ECC_INT_BASE ,
} ;
static struct hb_mc_settings mw_settings = {
. err_offset = MW_DDR_ECC_ERR_BASE ,
. int_offset = MW_DDR_ECC_INT_BASE ,
} ;
static struct of_device_id hb_ddr_ctrl_of_match [ ] = {
{ . compatible = " calxeda,hb-ddr-ctrl " , . data = & hb_settings } ,
{ . compatible = " calxeda,ecx-2000-ddr-ctrl " , . data = & mw_settings } ,
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , hb_ddr_ctrl_of_match ) ;
2012-12-22 01:23:51 +04:00
static int highbank_mc_probe ( struct platform_device * pdev )
2012-06-13 21:01:55 +04:00
{
2013-10-10 19:32:30 +04:00
const struct of_device_id * id ;
const struct hb_mc_settings * settings ;
2012-06-13 21:01:55 +04:00
struct edac_mc_layer layers [ 2 ] ;
struct mem_ctl_info * mci ;
struct hb_mc_drvdata * drvdata ;
struct dimm_info * dimm ;
struct resource * r ;
2013-10-10 19:32:30 +04:00
void __iomem * base ;
2012-06-13 21:01:55 +04:00
u32 control ;
int irq ;
int res = 0 ;
2013-10-10 19:32:30 +04:00
id = of_match_device ( hb_ddr_ctrl_of_match , & pdev - > dev ) ;
if ( ! id )
return - ENODEV ;
2012-06-13 21:01:55 +04:00
layers [ 0 ] . type = EDAC_MC_LAYER_CHIP_SELECT ;
layers [ 0 ] . size = 1 ;
layers [ 0 ] . is_virt_csrow = true ;
layers [ 1 ] . type = EDAC_MC_LAYER_CHANNEL ;
layers [ 1 ] . size = 1 ;
layers [ 1 ] . is_virt_csrow = false ;
mci = edac_mc_alloc ( 0 , ARRAY_SIZE ( layers ) , layers ,
sizeof ( struct hb_mc_drvdata ) ) ;
if ( ! mci )
return - ENOMEM ;
mci - > pdev = & pdev - > dev ;
drvdata = mci - > pvt_info ;
platform_set_drvdata ( pdev , mci ) ;
if ( ! devres_open_group ( & pdev - > dev , NULL , GFP_KERNEL ) )
return - ENOMEM ;
r = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( ! r ) {
dev_err ( & pdev - > dev , " Unable to get mem resource \n " ) ;
res = - ENODEV ;
goto err ;
}
if ( ! devm_request_mem_region ( & pdev - > dev , r - > start ,
resource_size ( r ) , dev_name ( & pdev - > dev ) ) ) {
dev_err ( & pdev - > dev , " Error while requesting mem region \n " ) ;
res = - EBUSY ;
goto err ;
}
2013-10-10 19:32:30 +04:00
base = devm_ioremap ( & pdev - > dev , r - > start , resource_size ( r ) ) ;
if ( ! base ) {
2012-06-13 21:01:55 +04:00
dev_err ( & pdev - > dev , " Unable to map regs \n " ) ;
res = - ENOMEM ;
goto err ;
}
2013-10-10 19:32:30 +04:00
settings = id - > data ;
drvdata - > mc_err_base = base + settings - > err_offset ;
drvdata - > mc_int_base = base + settings - > int_offset ;
control = readl ( drvdata - > mc_err_base + HB_DDR_ECC_OPT ) & 0x3 ;
2012-06-13 21:01:55 +04:00
if ( ! control | | ( control = = 0x2 ) ) {
dev_err ( & pdev - > dev , " No ECC present, or ECC disabled \n " ) ;
res = - ENODEV ;
goto err ;
}
mci - > mtype_cap = MEM_FLAG_DDR3 ;
mci - > edac_ctl_cap = EDAC_FLAG_NONE | EDAC_FLAG_SECDED ;
mci - > edac_cap = EDAC_FLAG_SECDED ;
2013-10-10 19:36:19 +04:00
mci - > mod_name = pdev - > dev . driver - > name ;
2012-06-13 21:01:55 +04:00
mci - > mod_ver = " 1 " ;
2013-10-10 19:36:19 +04:00
mci - > ctl_name = id - > compatible ;
mci - > dev_name = dev_name ( & pdev - > dev ) ;
2012-06-13 21:01:55 +04:00
mci - > scrub_mode = SCRUB_SW_SRC ;
/* Only a single 4GB DIMM is supported */
dimm = * mci - > dimms ;
dimm - > nr_pages = ( ~ 0UL > > PAGE_SHIFT ) + 1 ;
dimm - > grain = 8 ;
dimm - > dtype = DEV_X8 ;
dimm - > mtype = MEM_DDR3 ;
dimm - > edac_mode = EDAC_SECDED ;
res = edac_mc_add_mc ( mci ) ;
if ( res < 0 )
goto err ;
2013-10-10 20:23:38 +04:00
irq = platform_get_irq ( pdev , 0 ) ;
res = devm_request_irq ( & pdev - > dev , irq , highbank_mc_err_handler ,
0 , dev_name ( & pdev - > dev ) , mci ) ;
if ( res < 0 ) {
dev_err ( & pdev - > dev , " Unable to request irq %d \n " , irq ) ;
goto err2 ;
}
2012-06-13 21:01:55 +04:00
highbank_mc_create_debugfs_nodes ( mci ) ;
devres_close_group ( & pdev - > dev , NULL ) ;
return 0 ;
2013-10-10 20:23:38 +04:00
err2 :
edac_mc_del_mc ( & pdev - > dev ) ;
2012-06-13 21:01:55 +04:00
err :
devres_release_group ( & pdev - > dev , NULL ) ;
edac_mc_free ( mci ) ;
return res ;
}
static int highbank_mc_remove ( struct platform_device * pdev )
{
struct mem_ctl_info * mci = platform_get_drvdata ( pdev ) ;
edac_mc_del_mc ( & pdev - > dev ) ;
edac_mc_free ( mci ) ;
return 0 ;
}
static struct platform_driver highbank_mc_edac_driver = {
. probe = highbank_mc_probe ,
. remove = highbank_mc_remove ,
. driver = {
. name = " hb_mc_edac " ,
. of_match_table = hb_ddr_ctrl_of_match ,
} ,
} ;
module_platform_driver ( highbank_mc_edac_driver ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_AUTHOR ( " Calxeda, Inc. " ) ;
MODULE_DESCRIPTION ( " EDAC Driver for Calxeda Highbank " ) ;