2019-06-03 07:44:50 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2015-09-28 11:13:26 +02:00
/*
* Driver for Atmel Flexcom
*
* Copyright ( C ) 2015 Atmel Corporation
*
* Author : Cyrille Pitchen < cyrille . pitchen @ atmel . com >
*/
# include <linux/module.h>
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/platform_device.h>
# include <linux/of.h>
# include <linux/of_platform.h>
# include <linux/err.h>
# include <linux/io.h>
# include <linux/clk.h>
# include <dt-bindings/mfd/atmel-flexcom.h>
/* I/O register offsets */
# define FLEX_MR 0x0 /* Mode Register */
# define FLEX_VERSION 0xfc /* Version Register */
/* Mode Register bit fields */
# define FLEX_MR_OPMODE_OFFSET (0) /* Operating Mode */
# define FLEX_MR_OPMODE_MASK (0x3 << FLEX_MR_OPMODE_OFFSET)
# define FLEX_MR_OPMODE(opmode) (((opmode) << FLEX_MR_OPMODE_OFFSET) & \
FLEX_MR_OPMODE_MASK )
2017-12-12 17:21:19 +01:00
struct atmel_flexcom {
void __iomem * base ;
u32 opmode ;
struct clk * clk ;
} ;
2015-09-28 11:13:26 +02:00
static int atmel_flexcom_probe ( struct platform_device * pdev )
{
struct device_node * np = pdev - > dev . of_node ;
struct resource * res ;
2017-12-12 17:21:19 +01:00
struct atmel_flexcom * ddata ;
2015-09-28 11:13:26 +02:00
int err ;
2017-12-12 17:21:19 +01:00
ddata = devm_kzalloc ( & pdev - > dev , sizeof ( * ddata ) , GFP_KERNEL ) ;
if ( ! ddata )
return - ENOMEM ;
platform_set_drvdata ( pdev , ddata ) ;
err = of_property_read_u32 ( np , " atmel,flexcom-mode " , & ddata - > opmode ) ;
2015-09-28 11:13:26 +02:00
if ( err )
return err ;
2017-12-12 17:21:19 +01:00
if ( ddata - > opmode < ATMEL_FLEXCOM_MODE_USART | |
ddata - > opmode > ATMEL_FLEXCOM_MODE_TWI )
2015-09-28 11:13:26 +02:00
return - EINVAL ;
res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
2017-12-12 17:21:19 +01:00
ddata - > base = devm_ioremap_resource ( & pdev - > dev , res ) ;
if ( IS_ERR ( ddata - > base ) )
return PTR_ERR ( ddata - > base ) ;
2015-09-28 11:13:26 +02:00
2017-12-12 17:21:19 +01:00
ddata - > clk = devm_clk_get ( & pdev - > dev , NULL ) ;
if ( IS_ERR ( ddata - > clk ) )
return PTR_ERR ( ddata - > clk ) ;
2015-09-28 11:13:26 +02:00
2017-12-12 17:21:19 +01:00
err = clk_prepare_enable ( ddata - > clk ) ;
2015-09-28 11:13:26 +02:00
if ( err )
return err ;
/*
* Set the Operating Mode in the Mode Register : only the selected device
* is clocked . Hence , registers of the other serial devices remain
* inaccessible and are read as zero . Also the external I / O lines of the
* Flexcom are muxed to reach the selected device .
*/
2017-12-12 17:21:19 +01:00
writel ( FLEX_MR_OPMODE ( ddata - > opmode ) , ddata - > base + FLEX_MR ) ;
2015-09-28 11:13:26 +02:00
2017-12-12 17:21:19 +01:00
clk_disable_unprepare ( ddata - > clk ) ;
2015-09-28 11:13:26 +02:00
2017-05-29 17:45:54 +02:00
return devm_of_platform_populate ( & pdev - > dev ) ;
2015-09-28 11:13:26 +02:00
}
static const struct of_device_id atmel_flexcom_of_match [ ] = {
{ . compatible = " atmel,sama5d2-flexcom " } ,
{ /* sentinel */ }
} ;
MODULE_DEVICE_TABLE ( of , atmel_flexcom_of_match ) ;
2017-12-12 17:21:19 +01:00
# ifdef CONFIG_PM_SLEEP
static int atmel_flexcom_resume ( struct device * dev )
{
struct atmel_flexcom * ddata = dev_get_drvdata ( dev ) ;
int err ;
u32 val ;
err = clk_prepare_enable ( ddata - > clk ) ;
if ( err )
return err ;
val = FLEX_MR_OPMODE ( ddata - > opmode ) ,
writel ( val , ddata - > base + FLEX_MR ) ;
clk_disable_unprepare ( ddata - > clk ) ;
return 0 ;
}
# endif
static SIMPLE_DEV_PM_OPS ( atmel_flexcom_pm_ops , NULL ,
atmel_flexcom_resume ) ;
2015-09-28 11:13:26 +02:00
static struct platform_driver atmel_flexcom_driver = {
. probe = atmel_flexcom_probe ,
. driver = {
. name = " atmel_flexcom " ,
2017-12-12 17:21:19 +01:00
. pm = & atmel_flexcom_pm_ops ,
2015-09-28 11:13:26 +02:00
. of_match_table = atmel_flexcom_of_match ,
} ,
} ;
module_platform_driver ( atmel_flexcom_driver ) ;
MODULE_AUTHOR ( " Cyrille Pitchen <cyrille.pitchen@atmel.com> " ) ;
MODULE_DESCRIPTION ( " Atmel Flexcom MFD driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;