162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * sl3516-ce-rng.c - hardware cryptographic offloader for SL3516 SoC.
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright (C) 2021 Corentin Labbe <clabbe@baylibre.com>
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * This file handle the RNG found in the SL3516 crypto engine
862306a36Sopenharmony_ci */
962306a36Sopenharmony_ci#include "sl3516-ce.h"
1062306a36Sopenharmony_ci#include <linux/pm_runtime.h>
1162306a36Sopenharmony_ci#include <linux/hw_random.h>
1262306a36Sopenharmony_ci
1362306a36Sopenharmony_cistatic int sl3516_ce_rng_read(struct hwrng *rng, void *buf, size_t max, bool wait)
1462306a36Sopenharmony_ci{
1562306a36Sopenharmony_ci	struct sl3516_ce_dev *ce;
1662306a36Sopenharmony_ci	u32 *data = buf;
1762306a36Sopenharmony_ci	size_t read = 0;
1862306a36Sopenharmony_ci	int err;
1962306a36Sopenharmony_ci
2062306a36Sopenharmony_ci	ce = container_of(rng, struct sl3516_ce_dev, trng);
2162306a36Sopenharmony_ci
2262306a36Sopenharmony_ci#ifdef CONFIG_CRYPTO_DEV_SL3516_DEBUG
2362306a36Sopenharmony_ci	ce->hwrng_stat_req++;
2462306a36Sopenharmony_ci	ce->hwrng_stat_bytes += max;
2562306a36Sopenharmony_ci#endif
2662306a36Sopenharmony_ci
2762306a36Sopenharmony_ci	err = pm_runtime_get_sync(ce->dev);
2862306a36Sopenharmony_ci	if (err < 0) {
2962306a36Sopenharmony_ci		pm_runtime_put_noidle(ce->dev);
3062306a36Sopenharmony_ci		return err;
3162306a36Sopenharmony_ci	}
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_ci	while (read < max) {
3462306a36Sopenharmony_ci		*data = readl(ce->base + IPSEC_RAND_NUM_REG);
3562306a36Sopenharmony_ci		data++;
3662306a36Sopenharmony_ci		read += 4;
3762306a36Sopenharmony_ci	}
3862306a36Sopenharmony_ci
3962306a36Sopenharmony_ci	pm_runtime_put(ce->dev);
4062306a36Sopenharmony_ci
4162306a36Sopenharmony_ci	return read;
4262306a36Sopenharmony_ci}
4362306a36Sopenharmony_ci
4462306a36Sopenharmony_ciint sl3516_ce_rng_register(struct sl3516_ce_dev *ce)
4562306a36Sopenharmony_ci{
4662306a36Sopenharmony_ci	int ret;
4762306a36Sopenharmony_ci
4862306a36Sopenharmony_ci	ce->trng.name = "SL3516 Crypto Engine RNG";
4962306a36Sopenharmony_ci	ce->trng.read = sl3516_ce_rng_read;
5062306a36Sopenharmony_ci	ce->trng.quality = 700;
5162306a36Sopenharmony_ci
5262306a36Sopenharmony_ci	ret = hwrng_register(&ce->trng);
5362306a36Sopenharmony_ci	if (ret)
5462306a36Sopenharmony_ci		dev_err(ce->dev, "Fail to register the RNG\n");
5562306a36Sopenharmony_ci	return ret;
5662306a36Sopenharmony_ci}
5762306a36Sopenharmony_ci
5862306a36Sopenharmony_civoid sl3516_ce_rng_unregister(struct sl3516_ce_dev *ce)
5962306a36Sopenharmony_ci{
6062306a36Sopenharmony_ci	hwrng_unregister(&ce->trng);
6162306a36Sopenharmony_ci}
62