2019-09-12 12:01:49 +03:00
// SPDX-License-Identifier: GPL-2.0
// Copyright (c) 2019 Nuvoton Technology corporation.
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/io.h>
# include <linux/iopoll.h>
# include <linux/init.h>
# include <linux/random.h>
# include <linux/err.h>
2023-07-28 16:48:27 +03:00
# include <linux/of.h>
2019-09-12 12:01:49 +03:00
# include <linux/platform_device.h>
# include <linux/hw_random.h>
# include <linux/delay.h>
# include <linux/pm_runtime.h>
# define NPCM_RNGCS_REG 0x00 /* Control and status register */
# define NPCM_RNGD_REG 0x04 /* Data register */
# define NPCM_RNGMODE_REG 0x08 /* Mode register */
2022-09-29 16:31:11 +03:00
# define NPCM_RNG_CLK_SET_62_5MHZ BIT(2) /* 60-80 MHz */
2019-09-12 12:01:49 +03:00
# define NPCM_RNG_CLK_SET_25MHZ GENMASK(4, 3) /* 20-25 MHz */
# define NPCM_RNG_DATA_VALID BIT(1)
# define NPCM_RNG_ENABLE BIT(0)
# define NPCM_RNG_M1ROSEL BIT(1)
# define NPCM_RNG_TIMEOUT_USEC 20000
# define NPCM_RNG_POLL_USEC 1000
# define to_npcm_rng(p) container_of(p, struct npcm_rng, rng)
struct npcm_rng {
void __iomem * base ;
struct hwrng rng ;
2022-09-29 16:31:11 +03:00
u32 clkp ;
2019-09-12 12:01:49 +03:00
} ;
static int npcm_rng_init ( struct hwrng * rng )
{
struct npcm_rng * priv = to_npcm_rng ( rng ) ;
2022-09-29 16:31:11 +03:00
writel ( priv - > clkp | NPCM_RNG_ENABLE , priv - > base + NPCM_RNGCS_REG ) ;
2019-09-12 12:01:49 +03:00
return 0 ;
}
static void npcm_rng_cleanup ( struct hwrng * rng )
{
struct npcm_rng * priv = to_npcm_rng ( rng ) ;
2022-09-29 16:31:11 +03:00
writel ( priv - > clkp , priv - > base + NPCM_RNGCS_REG ) ;
2019-09-12 12:01:49 +03:00
}
static int npcm_rng_read ( struct hwrng * rng , void * buf , size_t max , bool wait )
{
struct npcm_rng * priv = to_npcm_rng ( rng ) ;
int retval = 0 ;
int ready ;
pm_runtime_get_sync ( ( struct device * ) priv - > rng . priv ) ;
2020-09-24 00:23:05 +03:00
while ( max ) {
2019-09-12 12:01:49 +03:00
if ( wait ) {
2020-09-24 00:23:05 +03:00
if ( readb_poll_timeout ( priv - > base + NPCM_RNGCS_REG ,
2019-09-12 12:01:49 +03:00
ready ,
ready & NPCM_RNG_DATA_VALID ,
NPCM_RNG_POLL_USEC ,
NPCM_RNG_TIMEOUT_USEC ) )
break ;
} else {
2020-09-24 00:23:05 +03:00
if ( ( readb ( priv - > base + NPCM_RNGCS_REG ) &
2019-09-12 12:01:49 +03:00
NPCM_RNG_DATA_VALID ) = = 0 )
break ;
}
2020-09-24 00:23:05 +03:00
* ( u8 * ) buf = readb ( priv - > base + NPCM_RNGD_REG ) ;
retval + + ;
buf + + ;
max - - ;
2019-09-12 12:01:49 +03:00
}
pm_runtime_mark_last_busy ( ( struct device * ) priv - > rng . priv ) ;
pm_runtime_put_sync_autosuspend ( ( struct device * ) priv - > rng . priv ) ;
return retval | | ! wait ? retval : - EIO ;
}
static int npcm_rng_probe ( struct platform_device * pdev )
{
struct npcm_rng * priv ;
int ret ;
priv = devm_kzalloc ( & pdev - > dev , sizeof ( * priv ) , GFP_KERNEL ) ;
if ( ! priv )
return - ENOMEM ;
2019-10-16 13:46:15 +03:00
priv - > base = devm_platform_ioremap_resource ( pdev , 0 ) ;
2019-09-12 12:01:49 +03:00
if ( IS_ERR ( priv - > base ) )
return PTR_ERR ( priv - > base ) ;
dev_set_drvdata ( & pdev - > dev , priv ) ;
pm_runtime_set_autosuspend_delay ( & pdev - > dev , 100 ) ;
pm_runtime_use_autosuspend ( & pdev - > dev ) ;
pm_runtime_enable ( & pdev - > dev ) ;
# ifndef CONFIG_PM
priv - > rng . init = npcm_rng_init ;
priv - > rng . cleanup = npcm_rng_cleanup ;
# endif
priv - > rng . name = pdev - > name ;
priv - > rng . read = npcm_rng_read ;
priv - > rng . priv = ( unsigned long ) & pdev - > dev ;
2022-09-29 16:31:11 +03:00
priv - > clkp = ( u32 ) ( uintptr_t ) of_device_get_match_data ( & pdev - > dev ) ;
2019-09-12 12:01:49 +03:00
writel ( NPCM_RNG_M1ROSEL , priv - > base + NPCM_RNGMODE_REG ) ;
ret = devm_hwrng_register ( & pdev - > dev , & priv - > rng ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " Failed to register rng device: %d \n " ,
ret ) ;
pm_runtime_disable ( & pdev - > dev ) ;
pm_runtime_set_suspended ( & pdev - > dev ) ;
return ret ;
}
return 0 ;
}
2023-12-11 01:12:23 +03:00
static void npcm_rng_remove ( struct platform_device * pdev )
2019-09-12 12:01:49 +03:00
{
struct npcm_rng * priv = platform_get_drvdata ( pdev ) ;
devm_hwrng_unregister ( & pdev - > dev , & priv - > rng ) ;
pm_runtime_disable ( & pdev - > dev ) ;
pm_runtime_set_suspended ( & pdev - > dev ) ;
}
# ifdef CONFIG_PM
static int npcm_rng_runtime_suspend ( struct device * dev )
{
struct npcm_rng * priv = dev_get_drvdata ( dev ) ;
npcm_rng_cleanup ( & priv - > rng ) ;
return 0 ;
}
static int npcm_rng_runtime_resume ( struct device * dev )
{
struct npcm_rng * priv = dev_get_drvdata ( dev ) ;
return npcm_rng_init ( & priv - > rng ) ;
}
# endif
static const struct dev_pm_ops npcm_rng_pm_ops = {
SET_RUNTIME_PM_OPS ( npcm_rng_runtime_suspend ,
npcm_rng_runtime_resume , NULL )
SET_SYSTEM_SLEEP_PM_OPS ( pm_runtime_force_suspend ,
pm_runtime_force_resume )
} ;
2020-06-29 11:03:53 +03:00
static const struct of_device_id rng_dt_id [ ] __maybe_unused = {
2022-09-29 16:31:11 +03:00
{ . compatible = " nuvoton,npcm750-rng " ,
. data = ( void * ) NPCM_RNG_CLK_SET_25MHZ } ,
{ . compatible = " nuvoton,npcm845-rng " ,
. data = ( void * ) NPCM_RNG_CLK_SET_62_5MHZ } ,
2019-09-12 12:01:49 +03:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , rng_dt_id ) ;
static struct platform_driver npcm_rng_driver = {
. driver = {
. name = " npcm-rng " ,
. pm = & npcm_rng_pm_ops ,
. of_match_table = of_match_ptr ( rng_dt_id ) ,
} ,
. probe = npcm_rng_probe ,
2023-12-11 01:12:23 +03:00
. remove_new = npcm_rng_remove ,
2019-09-12 12:01:49 +03:00
} ;
module_platform_driver ( npcm_rng_driver ) ;
MODULE_DESCRIPTION ( " Nuvoton NPCM Random Number Generator Driver " ) ;
MODULE_AUTHOR ( " Tomer Maimon <tomer.maimon@nuvoton.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;