2015-07-27 14:15:00 +03:00
/*
* Copyright ( C ) 2015 Srinivas Kandagatla < srinivas . kandagatla @ linaro . org >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 and
* only version 2 as published by the Free Software Foundation .
*
* 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 .
*/
# include <linux/device.h>
# include <linux/module.h>
2016-04-24 22:28:08 +03:00
# include <linux/io.h>
2015-07-27 14:15:00 +03:00
# include <linux/nvmem-provider.h>
# include <linux/platform_device.h>
2017-10-20 19:57:41 +03:00
struct qfprom_priv {
void __iomem * base ;
} ;
2016-04-24 22:28:08 +03:00
static int qfprom_reg_read ( void * context ,
unsigned int reg , void * _val , size_t bytes )
{
2017-10-20 19:57:41 +03:00
struct qfprom_priv * priv = context ;
2017-01-04 19:18:09 +03:00
u8 * val = _val ;
int i = 0 , words = bytes ;
2015-07-27 14:15:00 +03:00
2016-04-24 22:28:08 +03:00
while ( words - - )
2017-10-20 19:57:41 +03:00
* val + + = readb ( priv - > base + reg + i + + ) ;
2016-04-24 22:28:08 +03:00
return 0 ;
}
static int qfprom_reg_write ( void * context ,
unsigned int reg , void * _val , size_t bytes )
{
2017-10-20 19:57:41 +03:00
struct qfprom_priv * priv = context ;
2017-01-04 19:18:09 +03:00
u8 * val = _val ;
int i = 0 , words = bytes ;
2016-04-24 22:28:08 +03:00
while ( words - - )
2017-10-20 19:57:41 +03:00
writeb ( * val + + , priv - > base + reg + i + + ) ;
2016-04-24 22:28:08 +03:00
return 0 ;
}
2015-07-27 14:15:00 +03:00
static int qfprom_remove ( struct platform_device * pdev )
{
struct nvmem_device * nvmem = platform_get_drvdata ( pdev ) ;
return nvmem_unregister ( nvmem ) ;
}
2016-04-24 22:28:08 +03:00
static struct nvmem_config econfig = {
. name = " qfprom " ,
2017-01-04 19:18:09 +03:00
. stride = 1 ,
2016-04-24 22:28:08 +03:00
. word_size = 1 ,
. reg_read = qfprom_reg_read ,
. reg_write = qfprom_reg_write ,
} ;
2015-07-27 14:15:00 +03:00
static int qfprom_probe ( struct platform_device * pdev )
{
struct device * dev = & pdev - > dev ;
struct resource * res ;
struct nvmem_device * nvmem ;
2017-10-20 19:57:41 +03:00
struct qfprom_priv * priv ;
priv = devm_kzalloc ( dev , sizeof ( * priv ) , GFP_KERNEL ) ;
if ( ! priv )
return - ENOMEM ;
2015-07-27 14:15:00 +03:00
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2017-10-20 19:57:41 +03:00
priv - > base = devm_ioremap_resource ( dev , res ) ;
if ( IS_ERR ( priv - > base ) )
return PTR_ERR ( priv - > base ) ;
2015-07-27 14:15:00 +03:00
2016-04-24 22:28:08 +03:00
econfig . size = resource_size ( res ) ;
2015-07-27 14:15:00 +03:00
econfig . dev = dev ;
2017-10-20 19:57:41 +03:00
econfig . priv = priv ;
2016-04-24 22:28:08 +03:00
2015-07-27 14:15:00 +03:00
nvmem = nvmem_register ( & econfig ) ;
if ( IS_ERR ( nvmem ) )
return PTR_ERR ( nvmem ) ;
platform_set_drvdata ( pdev , nvmem ) ;
return 0 ;
}
static const struct of_device_id qfprom_of_match [ ] = {
{ . compatible = " qcom,qfprom " , } ,
{ /* sentinel */ } ,
} ;
MODULE_DEVICE_TABLE ( of , qfprom_of_match ) ;
static struct platform_driver qfprom_driver = {
. probe = qfprom_probe ,
. remove = qfprom_remove ,
. driver = {
. name = " qcom,qfprom " ,
. of_match_table = qfprom_of_match ,
} ,
} ;
module_platform_driver ( qfprom_driver ) ;
MODULE_AUTHOR ( " Srinivas Kandagatla <srinivas.kandagatla@linaro.org> " ) ;
MODULE_DESCRIPTION ( " Qualcomm QFPROM driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;