18c2ecf20Sopenharmony_ci/*
28c2ecf20Sopenharmony_ci * omap-rng.c - RNG driver for TI OMAP CPU family
38c2ecf20Sopenharmony_ci *
48c2ecf20Sopenharmony_ci * Author: Deepak Saxena <dsaxena@plexity.net>
58c2ecf20Sopenharmony_ci *
68c2ecf20Sopenharmony_ci * Copyright 2005 (c) MontaVista Software, Inc.
78c2ecf20Sopenharmony_ci *
88c2ecf20Sopenharmony_ci * Mostly based on original driver:
98c2ecf20Sopenharmony_ci *
108c2ecf20Sopenharmony_ci * Copyright (C) 2005 Nokia Corporation
118c2ecf20Sopenharmony_ci * Author: Juha Yrjölä <juha.yrjola@nokia.com>
128c2ecf20Sopenharmony_ci *
138c2ecf20Sopenharmony_ci * This file is licensed under  the terms of the GNU General Public
148c2ecf20Sopenharmony_ci * License version 2. This program is licensed "as is" without any
158c2ecf20Sopenharmony_ci * warranty of any kind, whether express or implied.
168c2ecf20Sopenharmony_ci */
178c2ecf20Sopenharmony_ci
188c2ecf20Sopenharmony_ci#include <linux/module.h>
198c2ecf20Sopenharmony_ci#include <linux/init.h>
208c2ecf20Sopenharmony_ci#include <linux/random.h>
218c2ecf20Sopenharmony_ci#include <linux/err.h>
228c2ecf20Sopenharmony_ci#include <linux/platform_device.h>
238c2ecf20Sopenharmony_ci#include <linux/hw_random.h>
248c2ecf20Sopenharmony_ci#include <linux/delay.h>
258c2ecf20Sopenharmony_ci#include <linux/kernel.h>
268c2ecf20Sopenharmony_ci#include <linux/slab.h>
278c2ecf20Sopenharmony_ci#include <linux/pm_runtime.h>
288c2ecf20Sopenharmony_ci#include <linux/of.h>
298c2ecf20Sopenharmony_ci#include <linux/of_device.h>
308c2ecf20Sopenharmony_ci#include <linux/of_address.h>
318c2ecf20Sopenharmony_ci#include <linux/interrupt.h>
328c2ecf20Sopenharmony_ci#include <linux/clk.h>
338c2ecf20Sopenharmony_ci
348c2ecf20Sopenharmony_ci#include <asm/io.h>
358c2ecf20Sopenharmony_ci
368c2ecf20Sopenharmony_ci#define RNG_REG_STATUS_RDY			(1 << 0)
378c2ecf20Sopenharmony_ci
388c2ecf20Sopenharmony_ci#define RNG_REG_INTACK_RDY_MASK			(1 << 0)
398c2ecf20Sopenharmony_ci#define RNG_REG_INTACK_SHUTDOWN_OFLO_MASK	(1 << 1)
408c2ecf20Sopenharmony_ci#define RNG_SHUTDOWN_OFLO_MASK			(1 << 1)
418c2ecf20Sopenharmony_ci
428c2ecf20Sopenharmony_ci#define RNG_CONTROL_STARTUP_CYCLES_SHIFT	16
438c2ecf20Sopenharmony_ci#define RNG_CONTROL_STARTUP_CYCLES_MASK		(0xffff << 16)
448c2ecf20Sopenharmony_ci#define RNG_CONTROL_ENABLE_TRNG_SHIFT		10
458c2ecf20Sopenharmony_ci#define RNG_CONTROL_ENABLE_TRNG_MASK		(1 << 10)
468c2ecf20Sopenharmony_ci
478c2ecf20Sopenharmony_ci#define RNG_CONFIG_MAX_REFIL_CYCLES_SHIFT	16
488c2ecf20Sopenharmony_ci#define RNG_CONFIG_MAX_REFIL_CYCLES_MASK	(0xffff << 16)
498c2ecf20Sopenharmony_ci#define RNG_CONFIG_MIN_REFIL_CYCLES_SHIFT	0
508c2ecf20Sopenharmony_ci#define RNG_CONFIG_MIN_REFIL_CYCLES_MASK	(0xff << 0)
518c2ecf20Sopenharmony_ci
528c2ecf20Sopenharmony_ci#define RNG_CONTROL_STARTUP_CYCLES		0xff
538c2ecf20Sopenharmony_ci#define RNG_CONFIG_MIN_REFIL_CYCLES		0x21
548c2ecf20Sopenharmony_ci#define RNG_CONFIG_MAX_REFIL_CYCLES		0x22
558c2ecf20Sopenharmony_ci
568c2ecf20Sopenharmony_ci#define RNG_ALARMCNT_ALARM_TH_SHIFT		0x0
578c2ecf20Sopenharmony_ci#define RNG_ALARMCNT_ALARM_TH_MASK		(0xff << 0)
588c2ecf20Sopenharmony_ci#define RNG_ALARMCNT_SHUTDOWN_TH_SHIFT		16
598c2ecf20Sopenharmony_ci#define RNG_ALARMCNT_SHUTDOWN_TH_MASK		(0x1f << 16)
608c2ecf20Sopenharmony_ci#define RNG_ALARM_THRESHOLD			0xff
618c2ecf20Sopenharmony_ci#define RNG_SHUTDOWN_THRESHOLD			0x4
628c2ecf20Sopenharmony_ci
638c2ecf20Sopenharmony_ci#define RNG_REG_FROENABLE_MASK			0xffffff
648c2ecf20Sopenharmony_ci#define RNG_REG_FRODETUNE_MASK			0xffffff
658c2ecf20Sopenharmony_ci
668c2ecf20Sopenharmony_ci#define OMAP2_RNG_OUTPUT_SIZE			0x4
678c2ecf20Sopenharmony_ci#define OMAP4_RNG_OUTPUT_SIZE			0x8
688c2ecf20Sopenharmony_ci#define EIP76_RNG_OUTPUT_SIZE			0x10
698c2ecf20Sopenharmony_ci
708c2ecf20Sopenharmony_ci/*
718c2ecf20Sopenharmony_ci * EIP76 RNG takes approx. 700us to produce 16 bytes of output data
728c2ecf20Sopenharmony_ci * as per testing results. And to account for the lack of udelay()'s
738c2ecf20Sopenharmony_ci * reliability, we keep the timeout as 1000us.
748c2ecf20Sopenharmony_ci */
758c2ecf20Sopenharmony_ci#define RNG_DATA_FILL_TIMEOUT			100
768c2ecf20Sopenharmony_ci
778c2ecf20Sopenharmony_cienum {
788c2ecf20Sopenharmony_ci	RNG_OUTPUT_0_REG = 0,
798c2ecf20Sopenharmony_ci	RNG_OUTPUT_1_REG,
808c2ecf20Sopenharmony_ci	RNG_OUTPUT_2_REG,
818c2ecf20Sopenharmony_ci	RNG_OUTPUT_3_REG,
828c2ecf20Sopenharmony_ci	RNG_STATUS_REG,
838c2ecf20Sopenharmony_ci	RNG_INTMASK_REG,
848c2ecf20Sopenharmony_ci	RNG_INTACK_REG,
858c2ecf20Sopenharmony_ci	RNG_CONTROL_REG,
868c2ecf20Sopenharmony_ci	RNG_CONFIG_REG,
878c2ecf20Sopenharmony_ci	RNG_ALARMCNT_REG,
888c2ecf20Sopenharmony_ci	RNG_FROENABLE_REG,
898c2ecf20Sopenharmony_ci	RNG_FRODETUNE_REG,
908c2ecf20Sopenharmony_ci	RNG_ALARMMASK_REG,
918c2ecf20Sopenharmony_ci	RNG_ALARMSTOP_REG,
928c2ecf20Sopenharmony_ci	RNG_REV_REG,
938c2ecf20Sopenharmony_ci	RNG_SYSCONFIG_REG,
948c2ecf20Sopenharmony_ci};
958c2ecf20Sopenharmony_ci
968c2ecf20Sopenharmony_cistatic const u16 reg_map_omap2[] = {
978c2ecf20Sopenharmony_ci	[RNG_OUTPUT_0_REG]	= 0x0,
988c2ecf20Sopenharmony_ci	[RNG_STATUS_REG]	= 0x4,
998c2ecf20Sopenharmony_ci	[RNG_CONFIG_REG]	= 0x28,
1008c2ecf20Sopenharmony_ci	[RNG_REV_REG]		= 0x3c,
1018c2ecf20Sopenharmony_ci	[RNG_SYSCONFIG_REG]	= 0x40,
1028c2ecf20Sopenharmony_ci};
1038c2ecf20Sopenharmony_ci
1048c2ecf20Sopenharmony_cistatic const u16 reg_map_omap4[] = {
1058c2ecf20Sopenharmony_ci	[RNG_OUTPUT_0_REG]	= 0x0,
1068c2ecf20Sopenharmony_ci	[RNG_OUTPUT_1_REG]	= 0x4,
1078c2ecf20Sopenharmony_ci	[RNG_STATUS_REG]	= 0x8,
1088c2ecf20Sopenharmony_ci	[RNG_INTMASK_REG]	= 0xc,
1098c2ecf20Sopenharmony_ci	[RNG_INTACK_REG]	= 0x10,
1108c2ecf20Sopenharmony_ci	[RNG_CONTROL_REG]	= 0x14,
1118c2ecf20Sopenharmony_ci	[RNG_CONFIG_REG]	= 0x18,
1128c2ecf20Sopenharmony_ci	[RNG_ALARMCNT_REG]	= 0x1c,
1138c2ecf20Sopenharmony_ci	[RNG_FROENABLE_REG]	= 0x20,
1148c2ecf20Sopenharmony_ci	[RNG_FRODETUNE_REG]	= 0x24,
1158c2ecf20Sopenharmony_ci	[RNG_ALARMMASK_REG]	= 0x28,
1168c2ecf20Sopenharmony_ci	[RNG_ALARMSTOP_REG]	= 0x2c,
1178c2ecf20Sopenharmony_ci	[RNG_REV_REG]		= 0x1FE0,
1188c2ecf20Sopenharmony_ci	[RNG_SYSCONFIG_REG]	= 0x1FE4,
1198c2ecf20Sopenharmony_ci};
1208c2ecf20Sopenharmony_ci
1218c2ecf20Sopenharmony_cistatic const u16 reg_map_eip76[] = {
1228c2ecf20Sopenharmony_ci	[RNG_OUTPUT_0_REG]	= 0x0,
1238c2ecf20Sopenharmony_ci	[RNG_OUTPUT_1_REG]	= 0x4,
1248c2ecf20Sopenharmony_ci	[RNG_OUTPUT_2_REG]	= 0x8,
1258c2ecf20Sopenharmony_ci	[RNG_OUTPUT_3_REG]	= 0xc,
1268c2ecf20Sopenharmony_ci	[RNG_STATUS_REG]	= 0x10,
1278c2ecf20Sopenharmony_ci	[RNG_INTACK_REG]	= 0x10,
1288c2ecf20Sopenharmony_ci	[RNG_CONTROL_REG]	= 0x14,
1298c2ecf20Sopenharmony_ci	[RNG_CONFIG_REG]	= 0x18,
1308c2ecf20Sopenharmony_ci	[RNG_ALARMCNT_REG]	= 0x1c,
1318c2ecf20Sopenharmony_ci	[RNG_FROENABLE_REG]	= 0x20,
1328c2ecf20Sopenharmony_ci	[RNG_FRODETUNE_REG]	= 0x24,
1338c2ecf20Sopenharmony_ci	[RNG_ALARMMASK_REG]	= 0x28,
1348c2ecf20Sopenharmony_ci	[RNG_ALARMSTOP_REG]	= 0x2c,
1358c2ecf20Sopenharmony_ci	[RNG_REV_REG]		= 0x7c,
1368c2ecf20Sopenharmony_ci};
1378c2ecf20Sopenharmony_ci
1388c2ecf20Sopenharmony_cistruct omap_rng_dev;
1398c2ecf20Sopenharmony_ci/**
1408c2ecf20Sopenharmony_ci * struct omap_rng_pdata - RNG IP block-specific data
1418c2ecf20Sopenharmony_ci * @regs: Pointer to the register offsets structure.
1428c2ecf20Sopenharmony_ci * @data_size: No. of bytes in RNG output.
1438c2ecf20Sopenharmony_ci * @data_present: Callback to determine if data is available.
1448c2ecf20Sopenharmony_ci * @init: Callback for IP specific initialization sequence.
1458c2ecf20Sopenharmony_ci * @cleanup: Callback for IP specific cleanup sequence.
1468c2ecf20Sopenharmony_ci */
1478c2ecf20Sopenharmony_cistruct omap_rng_pdata {
1488c2ecf20Sopenharmony_ci	u16	*regs;
1498c2ecf20Sopenharmony_ci	u32	data_size;
1508c2ecf20Sopenharmony_ci	u32	(*data_present)(struct omap_rng_dev *priv);
1518c2ecf20Sopenharmony_ci	int	(*init)(struct omap_rng_dev *priv);
1528c2ecf20Sopenharmony_ci	void	(*cleanup)(struct omap_rng_dev *priv);
1538c2ecf20Sopenharmony_ci};
1548c2ecf20Sopenharmony_ci
1558c2ecf20Sopenharmony_cistruct omap_rng_dev {
1568c2ecf20Sopenharmony_ci	void __iomem			*base;
1578c2ecf20Sopenharmony_ci	struct device			*dev;
1588c2ecf20Sopenharmony_ci	const struct omap_rng_pdata	*pdata;
1598c2ecf20Sopenharmony_ci	struct hwrng rng;
1608c2ecf20Sopenharmony_ci	struct clk 			*clk;
1618c2ecf20Sopenharmony_ci	struct clk			*clk_reg;
1628c2ecf20Sopenharmony_ci};
1638c2ecf20Sopenharmony_ci
1648c2ecf20Sopenharmony_cistatic inline u32 omap_rng_read(struct omap_rng_dev *priv, u16 reg)
1658c2ecf20Sopenharmony_ci{
1668c2ecf20Sopenharmony_ci	return __raw_readl(priv->base + priv->pdata->regs[reg]);
1678c2ecf20Sopenharmony_ci}
1688c2ecf20Sopenharmony_ci
1698c2ecf20Sopenharmony_cistatic inline void omap_rng_write(struct omap_rng_dev *priv, u16 reg,
1708c2ecf20Sopenharmony_ci				      u32 val)
1718c2ecf20Sopenharmony_ci{
1728c2ecf20Sopenharmony_ci	__raw_writel(val, priv->base + priv->pdata->regs[reg]);
1738c2ecf20Sopenharmony_ci}
1748c2ecf20Sopenharmony_ci
1758c2ecf20Sopenharmony_ci
1768c2ecf20Sopenharmony_cistatic int omap_rng_do_read(struct hwrng *rng, void *data, size_t max,
1778c2ecf20Sopenharmony_ci			    bool wait)
1788c2ecf20Sopenharmony_ci{
1798c2ecf20Sopenharmony_ci	struct omap_rng_dev *priv;
1808c2ecf20Sopenharmony_ci	int i, present;
1818c2ecf20Sopenharmony_ci
1828c2ecf20Sopenharmony_ci	priv = (struct omap_rng_dev *)rng->priv;
1838c2ecf20Sopenharmony_ci
1848c2ecf20Sopenharmony_ci	if (max < priv->pdata->data_size)
1858c2ecf20Sopenharmony_ci		return 0;
1868c2ecf20Sopenharmony_ci
1878c2ecf20Sopenharmony_ci	for (i = 0; i < RNG_DATA_FILL_TIMEOUT; i++) {
1888c2ecf20Sopenharmony_ci		present = priv->pdata->data_present(priv);
1898c2ecf20Sopenharmony_ci		if (present || !wait)
1908c2ecf20Sopenharmony_ci			break;
1918c2ecf20Sopenharmony_ci
1928c2ecf20Sopenharmony_ci		udelay(10);
1938c2ecf20Sopenharmony_ci	}
1948c2ecf20Sopenharmony_ci	if (!present)
1958c2ecf20Sopenharmony_ci		return 0;
1968c2ecf20Sopenharmony_ci
1978c2ecf20Sopenharmony_ci	memcpy_fromio(data, priv->base + priv->pdata->regs[RNG_OUTPUT_0_REG],
1988c2ecf20Sopenharmony_ci		      priv->pdata->data_size);
1998c2ecf20Sopenharmony_ci
2008c2ecf20Sopenharmony_ci	if (priv->pdata->regs[RNG_INTACK_REG])
2018c2ecf20Sopenharmony_ci		omap_rng_write(priv, RNG_INTACK_REG, RNG_REG_INTACK_RDY_MASK);
2028c2ecf20Sopenharmony_ci
2038c2ecf20Sopenharmony_ci	return priv->pdata->data_size;
2048c2ecf20Sopenharmony_ci}
2058c2ecf20Sopenharmony_ci
2068c2ecf20Sopenharmony_cistatic int omap_rng_init(struct hwrng *rng)
2078c2ecf20Sopenharmony_ci{
2088c2ecf20Sopenharmony_ci	struct omap_rng_dev *priv;
2098c2ecf20Sopenharmony_ci
2108c2ecf20Sopenharmony_ci	priv = (struct omap_rng_dev *)rng->priv;
2118c2ecf20Sopenharmony_ci	return priv->pdata->init(priv);
2128c2ecf20Sopenharmony_ci}
2138c2ecf20Sopenharmony_ci
2148c2ecf20Sopenharmony_cistatic void omap_rng_cleanup(struct hwrng *rng)
2158c2ecf20Sopenharmony_ci{
2168c2ecf20Sopenharmony_ci	struct omap_rng_dev *priv;
2178c2ecf20Sopenharmony_ci
2188c2ecf20Sopenharmony_ci	priv = (struct omap_rng_dev *)rng->priv;
2198c2ecf20Sopenharmony_ci	priv->pdata->cleanup(priv);
2208c2ecf20Sopenharmony_ci}
2218c2ecf20Sopenharmony_ci
2228c2ecf20Sopenharmony_ci
2238c2ecf20Sopenharmony_cistatic inline u32 omap2_rng_data_present(struct omap_rng_dev *priv)
2248c2ecf20Sopenharmony_ci{
2258c2ecf20Sopenharmony_ci	return omap_rng_read(priv, RNG_STATUS_REG) ? 0 : 1;
2268c2ecf20Sopenharmony_ci}
2278c2ecf20Sopenharmony_ci
2288c2ecf20Sopenharmony_cistatic int omap2_rng_init(struct omap_rng_dev *priv)
2298c2ecf20Sopenharmony_ci{
2308c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_SYSCONFIG_REG, 0x1);
2318c2ecf20Sopenharmony_ci	return 0;
2328c2ecf20Sopenharmony_ci}
2338c2ecf20Sopenharmony_ci
2348c2ecf20Sopenharmony_cistatic void omap2_rng_cleanup(struct omap_rng_dev *priv)
2358c2ecf20Sopenharmony_ci{
2368c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_SYSCONFIG_REG, 0x0);
2378c2ecf20Sopenharmony_ci}
2388c2ecf20Sopenharmony_ci
2398c2ecf20Sopenharmony_cistatic struct omap_rng_pdata omap2_rng_pdata = {
2408c2ecf20Sopenharmony_ci	.regs		= (u16 *)reg_map_omap2,
2418c2ecf20Sopenharmony_ci	.data_size	= OMAP2_RNG_OUTPUT_SIZE,
2428c2ecf20Sopenharmony_ci	.data_present	= omap2_rng_data_present,
2438c2ecf20Sopenharmony_ci	.init		= omap2_rng_init,
2448c2ecf20Sopenharmony_ci	.cleanup	= omap2_rng_cleanup,
2458c2ecf20Sopenharmony_ci};
2468c2ecf20Sopenharmony_ci
2478c2ecf20Sopenharmony_cistatic inline u32 omap4_rng_data_present(struct omap_rng_dev *priv)
2488c2ecf20Sopenharmony_ci{
2498c2ecf20Sopenharmony_ci	return omap_rng_read(priv, RNG_STATUS_REG) & RNG_REG_STATUS_RDY;
2508c2ecf20Sopenharmony_ci}
2518c2ecf20Sopenharmony_ci
2528c2ecf20Sopenharmony_cistatic int eip76_rng_init(struct omap_rng_dev *priv)
2538c2ecf20Sopenharmony_ci{
2548c2ecf20Sopenharmony_ci	u32 val;
2558c2ecf20Sopenharmony_ci
2568c2ecf20Sopenharmony_ci	/* Return if RNG is already running. */
2578c2ecf20Sopenharmony_ci	if (omap_rng_read(priv, RNG_CONTROL_REG) & RNG_CONTROL_ENABLE_TRNG_MASK)
2588c2ecf20Sopenharmony_ci		return 0;
2598c2ecf20Sopenharmony_ci
2608c2ecf20Sopenharmony_ci	/*  Number of 512 bit blocks of raw Noise Source output data that must
2618c2ecf20Sopenharmony_ci	 *  be processed by either the Conditioning Function or the
2628c2ecf20Sopenharmony_ci	 *  SP 800-90 DRBG ‘BC_DF’ functionality to yield a ‘full entropy’
2638c2ecf20Sopenharmony_ci	 *  output value.
2648c2ecf20Sopenharmony_ci	 */
2658c2ecf20Sopenharmony_ci	val = 0x5 << RNG_CONFIG_MIN_REFIL_CYCLES_SHIFT;
2668c2ecf20Sopenharmony_ci
2678c2ecf20Sopenharmony_ci	/* Number of FRO samples that are XOR-ed together into one bit to be
2688c2ecf20Sopenharmony_ci	 * shifted into the main shift register
2698c2ecf20Sopenharmony_ci	 */
2708c2ecf20Sopenharmony_ci	val |= RNG_CONFIG_MAX_REFIL_CYCLES << RNG_CONFIG_MAX_REFIL_CYCLES_SHIFT;
2718c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_CONFIG_REG, val);
2728c2ecf20Sopenharmony_ci
2738c2ecf20Sopenharmony_ci	/* Enable all available FROs */
2748c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_FRODETUNE_REG, 0x0);
2758c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_FROENABLE_REG, RNG_REG_FROENABLE_MASK);
2768c2ecf20Sopenharmony_ci
2778c2ecf20Sopenharmony_ci	/* Enable TRNG */
2788c2ecf20Sopenharmony_ci	val = RNG_CONTROL_ENABLE_TRNG_MASK;
2798c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_CONTROL_REG, val);
2808c2ecf20Sopenharmony_ci
2818c2ecf20Sopenharmony_ci	return 0;
2828c2ecf20Sopenharmony_ci}
2838c2ecf20Sopenharmony_ci
2848c2ecf20Sopenharmony_cistatic int omap4_rng_init(struct omap_rng_dev *priv)
2858c2ecf20Sopenharmony_ci{
2868c2ecf20Sopenharmony_ci	u32 val;
2878c2ecf20Sopenharmony_ci
2888c2ecf20Sopenharmony_ci	/* Return if RNG is already running. */
2898c2ecf20Sopenharmony_ci	if (omap_rng_read(priv, RNG_CONTROL_REG) & RNG_CONTROL_ENABLE_TRNG_MASK)
2908c2ecf20Sopenharmony_ci		return 0;
2918c2ecf20Sopenharmony_ci
2928c2ecf20Sopenharmony_ci	val = RNG_CONFIG_MIN_REFIL_CYCLES << RNG_CONFIG_MIN_REFIL_CYCLES_SHIFT;
2938c2ecf20Sopenharmony_ci	val |= RNG_CONFIG_MAX_REFIL_CYCLES << RNG_CONFIG_MAX_REFIL_CYCLES_SHIFT;
2948c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_CONFIG_REG, val);
2958c2ecf20Sopenharmony_ci
2968c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_FRODETUNE_REG, 0x0);
2978c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_FROENABLE_REG, RNG_REG_FROENABLE_MASK);
2988c2ecf20Sopenharmony_ci	val = RNG_ALARM_THRESHOLD << RNG_ALARMCNT_ALARM_TH_SHIFT;
2998c2ecf20Sopenharmony_ci	val |= RNG_SHUTDOWN_THRESHOLD << RNG_ALARMCNT_SHUTDOWN_TH_SHIFT;
3008c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_ALARMCNT_REG, val);
3018c2ecf20Sopenharmony_ci
3028c2ecf20Sopenharmony_ci	val = RNG_CONTROL_STARTUP_CYCLES << RNG_CONTROL_STARTUP_CYCLES_SHIFT;
3038c2ecf20Sopenharmony_ci	val |= RNG_CONTROL_ENABLE_TRNG_MASK;
3048c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_CONTROL_REG, val);
3058c2ecf20Sopenharmony_ci
3068c2ecf20Sopenharmony_ci	return 0;
3078c2ecf20Sopenharmony_ci}
3088c2ecf20Sopenharmony_ci
3098c2ecf20Sopenharmony_cistatic void omap4_rng_cleanup(struct omap_rng_dev *priv)
3108c2ecf20Sopenharmony_ci{
3118c2ecf20Sopenharmony_ci	int val;
3128c2ecf20Sopenharmony_ci
3138c2ecf20Sopenharmony_ci	val = omap_rng_read(priv, RNG_CONTROL_REG);
3148c2ecf20Sopenharmony_ci	val &= ~RNG_CONTROL_ENABLE_TRNG_MASK;
3158c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_CONTROL_REG, val);
3168c2ecf20Sopenharmony_ci}
3178c2ecf20Sopenharmony_ci
3188c2ecf20Sopenharmony_cistatic irqreturn_t omap4_rng_irq(int irq, void *dev_id)
3198c2ecf20Sopenharmony_ci{
3208c2ecf20Sopenharmony_ci	struct omap_rng_dev *priv = dev_id;
3218c2ecf20Sopenharmony_ci	u32 fro_detune, fro_enable;
3228c2ecf20Sopenharmony_ci
3238c2ecf20Sopenharmony_ci	/*
3248c2ecf20Sopenharmony_ci	 * Interrupt raised by a fro shutdown threshold, do the following:
3258c2ecf20Sopenharmony_ci	 * 1. Clear the alarm events.
3268c2ecf20Sopenharmony_ci	 * 2. De tune the FROs which are shutdown.
3278c2ecf20Sopenharmony_ci	 * 3. Re enable the shutdown FROs.
3288c2ecf20Sopenharmony_ci	 */
3298c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_ALARMMASK_REG, 0x0);
3308c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_ALARMSTOP_REG, 0x0);
3318c2ecf20Sopenharmony_ci
3328c2ecf20Sopenharmony_ci	fro_enable = omap_rng_read(priv, RNG_FROENABLE_REG);
3338c2ecf20Sopenharmony_ci	fro_detune = ~fro_enable & RNG_REG_FRODETUNE_MASK;
3348c2ecf20Sopenharmony_ci	fro_detune = fro_detune | omap_rng_read(priv, RNG_FRODETUNE_REG);
3358c2ecf20Sopenharmony_ci	fro_enable = RNG_REG_FROENABLE_MASK;
3368c2ecf20Sopenharmony_ci
3378c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_FRODETUNE_REG, fro_detune);
3388c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_FROENABLE_REG, fro_enable);
3398c2ecf20Sopenharmony_ci
3408c2ecf20Sopenharmony_ci	omap_rng_write(priv, RNG_INTACK_REG, RNG_REG_INTACK_SHUTDOWN_OFLO_MASK);
3418c2ecf20Sopenharmony_ci
3428c2ecf20Sopenharmony_ci	return IRQ_HANDLED;
3438c2ecf20Sopenharmony_ci}
3448c2ecf20Sopenharmony_ci
3458c2ecf20Sopenharmony_cistatic struct omap_rng_pdata omap4_rng_pdata = {
3468c2ecf20Sopenharmony_ci	.regs		= (u16 *)reg_map_omap4,
3478c2ecf20Sopenharmony_ci	.data_size	= OMAP4_RNG_OUTPUT_SIZE,
3488c2ecf20Sopenharmony_ci	.data_present	= omap4_rng_data_present,
3498c2ecf20Sopenharmony_ci	.init		= omap4_rng_init,
3508c2ecf20Sopenharmony_ci	.cleanup	= omap4_rng_cleanup,
3518c2ecf20Sopenharmony_ci};
3528c2ecf20Sopenharmony_ci
3538c2ecf20Sopenharmony_cistatic struct omap_rng_pdata eip76_rng_pdata = {
3548c2ecf20Sopenharmony_ci	.regs		= (u16 *)reg_map_eip76,
3558c2ecf20Sopenharmony_ci	.data_size	= EIP76_RNG_OUTPUT_SIZE,
3568c2ecf20Sopenharmony_ci	.data_present	= omap4_rng_data_present,
3578c2ecf20Sopenharmony_ci	.init		= eip76_rng_init,
3588c2ecf20Sopenharmony_ci	.cleanup	= omap4_rng_cleanup,
3598c2ecf20Sopenharmony_ci};
3608c2ecf20Sopenharmony_ci
3618c2ecf20Sopenharmony_cistatic const struct of_device_id omap_rng_of_match[] __maybe_unused = {
3628c2ecf20Sopenharmony_ci		{
3638c2ecf20Sopenharmony_ci			.compatible	= "ti,omap2-rng",
3648c2ecf20Sopenharmony_ci			.data		= &omap2_rng_pdata,
3658c2ecf20Sopenharmony_ci		},
3668c2ecf20Sopenharmony_ci		{
3678c2ecf20Sopenharmony_ci			.compatible	= "ti,omap4-rng",
3688c2ecf20Sopenharmony_ci			.data		= &omap4_rng_pdata,
3698c2ecf20Sopenharmony_ci		},
3708c2ecf20Sopenharmony_ci		{
3718c2ecf20Sopenharmony_ci			.compatible	= "inside-secure,safexcel-eip76",
3728c2ecf20Sopenharmony_ci			.data		= &eip76_rng_pdata,
3738c2ecf20Sopenharmony_ci		},
3748c2ecf20Sopenharmony_ci		{},
3758c2ecf20Sopenharmony_ci};
3768c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(of, omap_rng_of_match);
3778c2ecf20Sopenharmony_ci
3788c2ecf20Sopenharmony_cistatic int of_get_omap_rng_device_details(struct omap_rng_dev *priv,
3798c2ecf20Sopenharmony_ci					  struct platform_device *pdev)
3808c2ecf20Sopenharmony_ci{
3818c2ecf20Sopenharmony_ci	const struct of_device_id *match;
3828c2ecf20Sopenharmony_ci	struct device *dev = &pdev->dev;
3838c2ecf20Sopenharmony_ci	int irq, err;
3848c2ecf20Sopenharmony_ci
3858c2ecf20Sopenharmony_ci	match = of_match_device(of_match_ptr(omap_rng_of_match), dev);
3868c2ecf20Sopenharmony_ci	if (!match) {
3878c2ecf20Sopenharmony_ci		dev_err(dev, "no compatible OF match\n");
3888c2ecf20Sopenharmony_ci		return -EINVAL;
3898c2ecf20Sopenharmony_ci	}
3908c2ecf20Sopenharmony_ci	priv->pdata = match->data;
3918c2ecf20Sopenharmony_ci
3928c2ecf20Sopenharmony_ci	if (of_device_is_compatible(dev->of_node, "ti,omap4-rng") ||
3938c2ecf20Sopenharmony_ci	    of_device_is_compatible(dev->of_node, "inside-secure,safexcel-eip76")) {
3948c2ecf20Sopenharmony_ci		irq = platform_get_irq(pdev, 0);
3958c2ecf20Sopenharmony_ci		if (irq < 0)
3968c2ecf20Sopenharmony_ci			return irq;
3978c2ecf20Sopenharmony_ci
3988c2ecf20Sopenharmony_ci		err = devm_request_irq(dev, irq, omap4_rng_irq,
3998c2ecf20Sopenharmony_ci				       IRQF_TRIGGER_NONE, dev_name(dev), priv);
4008c2ecf20Sopenharmony_ci		if (err) {
4018c2ecf20Sopenharmony_ci			dev_err(dev, "unable to request irq %d, err = %d\n",
4028c2ecf20Sopenharmony_ci				irq, err);
4038c2ecf20Sopenharmony_ci			return err;
4048c2ecf20Sopenharmony_ci		}
4058c2ecf20Sopenharmony_ci
4068c2ecf20Sopenharmony_ci		/*
4078c2ecf20Sopenharmony_ci		 * On OMAP4, enabling the shutdown_oflo interrupt is
4088c2ecf20Sopenharmony_ci		 * done in the interrupt mask register. There is no
4098c2ecf20Sopenharmony_ci		 * such register on EIP76, and it's enabled by the
4108c2ecf20Sopenharmony_ci		 * same bit in the control register
4118c2ecf20Sopenharmony_ci		 */
4128c2ecf20Sopenharmony_ci		if (priv->pdata->regs[RNG_INTMASK_REG])
4138c2ecf20Sopenharmony_ci			omap_rng_write(priv, RNG_INTMASK_REG,
4148c2ecf20Sopenharmony_ci				       RNG_SHUTDOWN_OFLO_MASK);
4158c2ecf20Sopenharmony_ci		else
4168c2ecf20Sopenharmony_ci			omap_rng_write(priv, RNG_CONTROL_REG,
4178c2ecf20Sopenharmony_ci				       RNG_SHUTDOWN_OFLO_MASK);
4188c2ecf20Sopenharmony_ci	}
4198c2ecf20Sopenharmony_ci	return 0;
4208c2ecf20Sopenharmony_ci}
4218c2ecf20Sopenharmony_ci
4228c2ecf20Sopenharmony_cistatic int get_omap_rng_device_details(struct omap_rng_dev *omap_rng)
4238c2ecf20Sopenharmony_ci{
4248c2ecf20Sopenharmony_ci	/* Only OMAP2/3 can be non-DT */
4258c2ecf20Sopenharmony_ci	omap_rng->pdata = &omap2_rng_pdata;
4268c2ecf20Sopenharmony_ci	return 0;
4278c2ecf20Sopenharmony_ci}
4288c2ecf20Sopenharmony_ci
4298c2ecf20Sopenharmony_cistatic int omap_rng_probe(struct platform_device *pdev)
4308c2ecf20Sopenharmony_ci{
4318c2ecf20Sopenharmony_ci	struct omap_rng_dev *priv;
4328c2ecf20Sopenharmony_ci	struct device *dev = &pdev->dev;
4338c2ecf20Sopenharmony_ci	int ret;
4348c2ecf20Sopenharmony_ci
4358c2ecf20Sopenharmony_ci	priv = devm_kzalloc(dev, sizeof(struct omap_rng_dev), GFP_KERNEL);
4368c2ecf20Sopenharmony_ci	if (!priv)
4378c2ecf20Sopenharmony_ci		return -ENOMEM;
4388c2ecf20Sopenharmony_ci
4398c2ecf20Sopenharmony_ci	priv->rng.read = omap_rng_do_read;
4408c2ecf20Sopenharmony_ci	priv->rng.init = omap_rng_init;
4418c2ecf20Sopenharmony_ci	priv->rng.cleanup = omap_rng_cleanup;
4428c2ecf20Sopenharmony_ci	priv->rng.quality = 900;
4438c2ecf20Sopenharmony_ci
4448c2ecf20Sopenharmony_ci	priv->rng.priv = (unsigned long)priv;
4458c2ecf20Sopenharmony_ci	platform_set_drvdata(pdev, priv);
4468c2ecf20Sopenharmony_ci	priv->dev = dev;
4478c2ecf20Sopenharmony_ci
4488c2ecf20Sopenharmony_ci	priv->base = devm_platform_ioremap_resource(pdev, 0);
4498c2ecf20Sopenharmony_ci	if (IS_ERR(priv->base)) {
4508c2ecf20Sopenharmony_ci		ret = PTR_ERR(priv->base);
4518c2ecf20Sopenharmony_ci		goto err_ioremap;
4528c2ecf20Sopenharmony_ci	}
4538c2ecf20Sopenharmony_ci
4548c2ecf20Sopenharmony_ci	priv->rng.name = devm_kstrdup(dev, dev_name(dev), GFP_KERNEL);
4558c2ecf20Sopenharmony_ci	if (!priv->rng.name) {
4568c2ecf20Sopenharmony_ci		ret = -ENOMEM;
4578c2ecf20Sopenharmony_ci		goto err_ioremap;
4588c2ecf20Sopenharmony_ci	}
4598c2ecf20Sopenharmony_ci
4608c2ecf20Sopenharmony_ci	pm_runtime_enable(&pdev->dev);
4618c2ecf20Sopenharmony_ci	ret = pm_runtime_get_sync(&pdev->dev);
4628c2ecf20Sopenharmony_ci	if (ret < 0) {
4638c2ecf20Sopenharmony_ci		dev_err(&pdev->dev, "Failed to runtime_get device: %d\n", ret);
4648c2ecf20Sopenharmony_ci		pm_runtime_put_noidle(&pdev->dev);
4658c2ecf20Sopenharmony_ci		goto err_ioremap;
4668c2ecf20Sopenharmony_ci	}
4678c2ecf20Sopenharmony_ci
4688c2ecf20Sopenharmony_ci	priv->clk = devm_clk_get(&pdev->dev, NULL);
4698c2ecf20Sopenharmony_ci	if (PTR_ERR(priv->clk) == -EPROBE_DEFER)
4708c2ecf20Sopenharmony_ci		return -EPROBE_DEFER;
4718c2ecf20Sopenharmony_ci	if (!IS_ERR(priv->clk)) {
4728c2ecf20Sopenharmony_ci		ret = clk_prepare_enable(priv->clk);
4738c2ecf20Sopenharmony_ci		if (ret) {
4748c2ecf20Sopenharmony_ci			dev_err(&pdev->dev,
4758c2ecf20Sopenharmony_ci				"Unable to enable the clk: %d\n", ret);
4768c2ecf20Sopenharmony_ci			goto err_register;
4778c2ecf20Sopenharmony_ci		}
4788c2ecf20Sopenharmony_ci	}
4798c2ecf20Sopenharmony_ci
4808c2ecf20Sopenharmony_ci	priv->clk_reg = devm_clk_get(&pdev->dev, "reg");
4818c2ecf20Sopenharmony_ci	if (PTR_ERR(priv->clk_reg) == -EPROBE_DEFER)
4828c2ecf20Sopenharmony_ci		return -EPROBE_DEFER;
4838c2ecf20Sopenharmony_ci	if (!IS_ERR(priv->clk_reg)) {
4848c2ecf20Sopenharmony_ci		ret = clk_prepare_enable(priv->clk_reg);
4858c2ecf20Sopenharmony_ci		if (ret) {
4868c2ecf20Sopenharmony_ci			dev_err(&pdev->dev,
4878c2ecf20Sopenharmony_ci				"Unable to enable the register clk: %d\n",
4888c2ecf20Sopenharmony_ci				ret);
4898c2ecf20Sopenharmony_ci			goto err_register;
4908c2ecf20Sopenharmony_ci		}
4918c2ecf20Sopenharmony_ci	}
4928c2ecf20Sopenharmony_ci
4938c2ecf20Sopenharmony_ci	ret = (dev->of_node) ? of_get_omap_rng_device_details(priv, pdev) :
4948c2ecf20Sopenharmony_ci				get_omap_rng_device_details(priv);
4958c2ecf20Sopenharmony_ci	if (ret)
4968c2ecf20Sopenharmony_ci		goto err_register;
4978c2ecf20Sopenharmony_ci
4988c2ecf20Sopenharmony_ci	ret = devm_hwrng_register(&pdev->dev, &priv->rng);
4998c2ecf20Sopenharmony_ci	if (ret)
5008c2ecf20Sopenharmony_ci		goto err_register;
5018c2ecf20Sopenharmony_ci
5028c2ecf20Sopenharmony_ci	dev_info(&pdev->dev, "Random Number Generator ver. %02x\n",
5038c2ecf20Sopenharmony_ci		 omap_rng_read(priv, RNG_REV_REG));
5048c2ecf20Sopenharmony_ci
5058c2ecf20Sopenharmony_ci	return 0;
5068c2ecf20Sopenharmony_ci
5078c2ecf20Sopenharmony_cierr_register:
5088c2ecf20Sopenharmony_ci	priv->base = NULL;
5098c2ecf20Sopenharmony_ci	pm_runtime_put_sync(&pdev->dev);
5108c2ecf20Sopenharmony_ci	pm_runtime_disable(&pdev->dev);
5118c2ecf20Sopenharmony_ci
5128c2ecf20Sopenharmony_ci	clk_disable_unprepare(priv->clk_reg);
5138c2ecf20Sopenharmony_ci	clk_disable_unprepare(priv->clk);
5148c2ecf20Sopenharmony_cierr_ioremap:
5158c2ecf20Sopenharmony_ci	dev_err(dev, "initialization failed.\n");
5168c2ecf20Sopenharmony_ci	return ret;
5178c2ecf20Sopenharmony_ci}
5188c2ecf20Sopenharmony_ci
5198c2ecf20Sopenharmony_cistatic int omap_rng_remove(struct platform_device *pdev)
5208c2ecf20Sopenharmony_ci{
5218c2ecf20Sopenharmony_ci	struct omap_rng_dev *priv = platform_get_drvdata(pdev);
5228c2ecf20Sopenharmony_ci
5238c2ecf20Sopenharmony_ci
5248c2ecf20Sopenharmony_ci	priv->pdata->cleanup(priv);
5258c2ecf20Sopenharmony_ci
5268c2ecf20Sopenharmony_ci	pm_runtime_put_sync(&pdev->dev);
5278c2ecf20Sopenharmony_ci	pm_runtime_disable(&pdev->dev);
5288c2ecf20Sopenharmony_ci
5298c2ecf20Sopenharmony_ci	clk_disable_unprepare(priv->clk);
5308c2ecf20Sopenharmony_ci	clk_disable_unprepare(priv->clk_reg);
5318c2ecf20Sopenharmony_ci
5328c2ecf20Sopenharmony_ci	return 0;
5338c2ecf20Sopenharmony_ci}
5348c2ecf20Sopenharmony_ci
5358c2ecf20Sopenharmony_cistatic int __maybe_unused omap_rng_suspend(struct device *dev)
5368c2ecf20Sopenharmony_ci{
5378c2ecf20Sopenharmony_ci	struct omap_rng_dev *priv = dev_get_drvdata(dev);
5388c2ecf20Sopenharmony_ci
5398c2ecf20Sopenharmony_ci	priv->pdata->cleanup(priv);
5408c2ecf20Sopenharmony_ci	pm_runtime_put_sync(dev);
5418c2ecf20Sopenharmony_ci
5428c2ecf20Sopenharmony_ci	return 0;
5438c2ecf20Sopenharmony_ci}
5448c2ecf20Sopenharmony_ci
5458c2ecf20Sopenharmony_cistatic int __maybe_unused omap_rng_resume(struct device *dev)
5468c2ecf20Sopenharmony_ci{
5478c2ecf20Sopenharmony_ci	struct omap_rng_dev *priv = dev_get_drvdata(dev);
5488c2ecf20Sopenharmony_ci	int ret;
5498c2ecf20Sopenharmony_ci
5508c2ecf20Sopenharmony_ci	ret = pm_runtime_get_sync(dev);
5518c2ecf20Sopenharmony_ci	if (ret < 0) {
5528c2ecf20Sopenharmony_ci		dev_err(dev, "Failed to runtime_get device: %d\n", ret);
5538c2ecf20Sopenharmony_ci		pm_runtime_put_noidle(dev);
5548c2ecf20Sopenharmony_ci		return ret;
5558c2ecf20Sopenharmony_ci	}
5568c2ecf20Sopenharmony_ci
5578c2ecf20Sopenharmony_ci	priv->pdata->init(priv);
5588c2ecf20Sopenharmony_ci
5598c2ecf20Sopenharmony_ci	return 0;
5608c2ecf20Sopenharmony_ci}
5618c2ecf20Sopenharmony_ci
5628c2ecf20Sopenharmony_cistatic SIMPLE_DEV_PM_OPS(omap_rng_pm, omap_rng_suspend, omap_rng_resume);
5638c2ecf20Sopenharmony_ci
5648c2ecf20Sopenharmony_cistatic struct platform_driver omap_rng_driver = {
5658c2ecf20Sopenharmony_ci	.driver = {
5668c2ecf20Sopenharmony_ci		.name		= "omap_rng",
5678c2ecf20Sopenharmony_ci		.pm		= &omap_rng_pm,
5688c2ecf20Sopenharmony_ci		.of_match_table = of_match_ptr(omap_rng_of_match),
5698c2ecf20Sopenharmony_ci	},
5708c2ecf20Sopenharmony_ci	.probe		= omap_rng_probe,
5718c2ecf20Sopenharmony_ci	.remove		= omap_rng_remove,
5728c2ecf20Sopenharmony_ci};
5738c2ecf20Sopenharmony_ci
5748c2ecf20Sopenharmony_cimodule_platform_driver(omap_rng_driver);
5758c2ecf20Sopenharmony_ciMODULE_ALIAS("platform:omap_rng");
5768c2ecf20Sopenharmony_ciMODULE_AUTHOR("Deepak Saxena (and others)");
5778c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL");
578