blob: d2479d244ed5ac578d69c24cb519f73545bd18c6 [file] [log] [blame]
// SPDX-License-Identifier: GPL-2.0
/*
* Copyright (c) 2024 Linaro Ltd.
* Author: Sam Protsenko <semen.protsenko@linaro.org>
*
* Samsung Exynos TRNG driver (True Random Number Generator).
*/
#include <clk.h>
#include <dm.h>
#include <rng.h>
#include <dm/device.h>
#include <dm/device_compat.h>
#include <asm/io.h>
#include <linux/arm-smccc.h>
#include <linux/bitops.h>
#include <linux/delay.h>
#include <linux/iopoll.h>
#include <linux/time.h>
#define EXYNOS_TRNG_CLKDIV 0x0
#define EXYNOS_TRNG_CLKDIV_MASK GENMASK(15, 0)
#define EXYNOS_TRNG_CLOCK_RATE 500000
#define EXYNOS_TRNG_CTRL 0x20
#define EXYNOS_TRNG_CTRL_RNGEN BIT(31)
#define EXYNOS_TRNG_POST_CTRL 0x30
#define EXYNOS_TRNG_ONLINE_CTRL 0x40
#define EXYNOS_TRNG_ONLINE_STAT 0x44
#define EXYNOS_TRNG_ONLINE_MAXCHI2 0x48
#define EXYNOS_TRNG_FIFO_CTRL 0x50
#define EXYNOS_TRNG_FIFO_0 0x80
#define EXYNOS_TRNG_FIFO_1 0x84
#define EXYNOS_TRNG_FIFO_2 0x88
#define EXYNOS_TRNG_FIFO_3 0x8c
#define EXYNOS_TRNG_FIFO_4 0x90
#define EXYNOS_TRNG_FIFO_5 0x94
#define EXYNOS_TRNG_FIFO_6 0x98
#define EXYNOS_TRNG_FIFO_7 0x9c
#define EXYNOS_TRNG_FIFO_LEN 8
#define EXYNOS_TRNG_FIFO_TIMEOUT (1 * USEC_PER_SEC)
#define EXYNOS_SMC_CALL_VAL(func_num) \
ARM_SMCCC_CALL_VAL(ARM_SMCCC_FAST_CALL, \
ARM_SMCCC_SMC_32, \
ARM_SMCCC_OWNER_SIP, \
func_num)
/* SMC command for DTRNG access */
#define SMC_CMD_RANDOM EXYNOS_SMC_CALL_VAL(0x1012)
/* SMC_CMD_RANDOM: arguments */
#define HWRNG_INIT 0x0
#define HWRNG_EXIT 0x1
#define HWRNG_GET_DATA 0x2
/* SMC_CMD_RANDOM: return values */
#define HWRNG_RET_OK 0x0
#define HWRNG_RET_RETRY_ERROR 0x2
#define HWRNG_MAX_TRIES 100
/**
* struct exynos_trng_variant - Chip specific data
*
* @smc: Set "true" if TRNG block has to be accessed via SMC calls
* @init: (Optional) TRNG initialization function to call on probe
* @exit: (Optional) TRNG deinitialization function to call on remove
* @read: Function to read the random data from TRNG block
*/
struct exynos_trng_variant {
bool smc;
int (*init)(struct udevice *dev);
void (*exit)(struct udevice *dev);
int (*read)(struct udevice *dev, void *data, size_t len);
};
/**
* struct exynos_trng_priv - Driver's private data
*
* @base: Base address of MMIO registers of TRNG block
* @clk: Operating clock (needed for TRNG block functioning)
* @pclk: Bus clock (needed for interfacing the TRNG block registers)
* @data: Chip specific data
*/
struct exynos_trng_priv {
void __iomem *base;
struct clk *clk;
struct clk *pclk;
const struct exynos_trng_variant *data;
};
static int exynos_trng_read_reg(struct udevice *dev, void *data, size_t len)
{
struct exynos_trng_priv *trng = dev_get_priv(dev);
int val;
len = min_t(size_t, len, EXYNOS_TRNG_FIFO_LEN * 4);
writel_relaxed(len * 8, trng->base + EXYNOS_TRNG_FIFO_CTRL);
val = readl_poll_timeout(trng->base + EXYNOS_TRNG_FIFO_CTRL, val,
val == 0, EXYNOS_TRNG_FIFO_TIMEOUT);
if (val < 0)
return val;
memcpy_fromio(data, trng->base + EXYNOS_TRNG_FIFO_0, len);
return 0;
}
static int exynos_trng_read_smc(struct udevice *dev, void *data, size_t len)
{
struct arm_smccc_res res;
unsigned int copied = 0;
u32 *buf = data;
int tries = 0;
while (copied < len) {
arm_smccc_smc(SMC_CMD_RANDOM, HWRNG_GET_DATA, 0, 0, 0, 0, 0, 0,
&res);
switch (res.a0) {
case HWRNG_RET_OK:
*buf++ = res.a2;
*buf++ = res.a3;
copied += 8;
tries = 0;
break;
case HWRNG_RET_RETRY_ERROR:
if (++tries >= HWRNG_MAX_TRIES)
return -EIO;
udelay(10);
break;
default:
return -EIO;
}
}
return 0;
}
static int exynos_trng_init_reg(struct udevice *dev)
{
const u32 max_div = EXYNOS_TRNG_CLKDIV_MASK;
struct exynos_trng_priv *trng = dev_get_priv(dev);
unsigned long sss_rate;
u32 div;
sss_rate = clk_get_rate(trng->clk);
/*
* For most TRNG circuits the clock frequency of under 500 kHz is safe.
* The clock divider should be an even number.
*/
div = sss_rate / EXYNOS_TRNG_CLOCK_RATE;
div -= div % 2; /* make sure it's even */
if (div > max_div) {
dev_err(dev, "Clock divider too large: %u", div);
return -ERANGE;
}
writel_relaxed(div, trng->base + EXYNOS_TRNG_CLKDIV);
/* Enable the generator */
writel_relaxed(EXYNOS_TRNG_CTRL_RNGEN, trng->base + EXYNOS_TRNG_CTRL);
/* Disable post-processing */
writel_relaxed(0, trng->base + EXYNOS_TRNG_POST_CTRL);
return 0;
}
static int exynos_trng_init_smc(struct udevice *dev)
{
struct arm_smccc_res res;
int ret = 0;
arm_smccc_smc(SMC_CMD_RANDOM, HWRNG_INIT, 0, 0, 0, 0, 0, 0, &res);
if (res.a0 != HWRNG_RET_OK) {
dev_err(dev, "SMC command for TRNG init failed (%d)\n",
(int)res.a0);
ret = -EIO;
}
if ((int)res.a0 == -1)
dev_info(dev, "Make sure LDFW is loaded\n");
return ret;
}
static void exynos_trng_exit_smc(struct udevice *dev)
{
struct arm_smccc_res res;
arm_smccc_smc(SMC_CMD_RANDOM, HWRNG_EXIT, 0, 0, 0, 0, 0, 0, &res);
}
static int exynos_trng_read(struct udevice *dev, void *data, size_t len)
{
struct exynos_trng_priv *trng = dev_get_priv(dev);
return trng->data->read(dev, data, len);
}
static int exynos_trng_of_to_plat(struct udevice *dev)
{
struct exynos_trng_priv *trng = dev_get_priv(dev);
trng->data = (struct exynos_trng_variant *)dev_get_driver_data(dev);
if (!trng->data->smc) {
trng->base = dev_read_addr_ptr(dev);
if (!trng->base)
return -EINVAL;
}
trng->clk = devm_clk_get(dev, "secss");
if (IS_ERR(trng->clk))
return PTR_ERR(trng->clk);
trng->pclk = devm_clk_get_optional(dev, "pclk");
if (IS_ERR(trng->pclk))
return PTR_ERR(trng->pclk);
return 0;
}
static int exynos_trng_probe(struct udevice *dev)
{
struct exynos_trng_priv *trng = dev_get_priv(dev);
int ret;
ret = clk_enable(trng->pclk);
if (ret)
return ret;
ret = clk_enable(trng->clk);
if (ret)
return ret;
if (trng->data->init)
ret = trng->data->init(dev);
return ret;
}
static int exynos_trng_remove(struct udevice *dev)
{
struct exynos_trng_priv *trng = dev_get_priv(dev);
if (trng->data->exit)
trng->data->exit(dev);
/* Keep SSS clocks enabled, they are needed for EL3_MON and kernel */
return 0;
}
static const struct dm_rng_ops exynos_trng_ops = {
.read = exynos_trng_read,
};
static const struct exynos_trng_variant exynos5250_trng_data = {
.init = exynos_trng_init_reg,
.read = exynos_trng_read_reg,
};
static const struct exynos_trng_variant exynos850_trng_data = {
.smc = true,
.init = exynos_trng_init_smc,
.exit = exynos_trng_exit_smc,
.read = exynos_trng_read_smc,
};
static const struct udevice_id exynos_trng_match[] = {
{
.compatible = "samsung,exynos5250-trng",
.data = (ulong)&exynos5250_trng_data,
}, {
.compatible = "samsung,exynos850-trng",
.data = (ulong)&exynos850_trng_data,
},
{ },
};
U_BOOT_DRIVER(exynos_trng) = {
.name = "exynos-trng",
.id = UCLASS_RNG,
.of_match = exynos_trng_match,
.of_to_plat = exynos_trng_of_to_plat,
.probe = exynos_trng_probe,
.remove = exynos_trng_remove,
.ops = &exynos_trng_ops,
.priv_auto = sizeof(struct exynos_trng_priv),
};