Files
thead-kernel/drivers/nvmem/light-efuse.c
thead_admin e17ac7bab2 Linux_SDK_V1.3.3
Signed-off-by: thead_admin <occ_thead@service.alibaba.com>
2024-01-31 22:09:29 +08:00

1167 lines
27 KiB
C

// SPDX-License-Identifier: GPL-2.0-only
/*
* Copyright (c) 2021 Alibaba Inc.
*/
#include <linux/clk.h>
#include <linux/compiler_types.h>
#include <linux/device.h>
#include <linux/pm_runtime.h>
#include <asm/io.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/mod_devicetable.h>
#include <linux/nvmem-provider.h>
#include <linux/platform_device.h>
#include <linux/regmap.h>
#include <linux/slab.h>
#include <linux/types.h>
#include <linux/mfd/syscon.h>
#define CON 0x00
#define LCPAR 0x04
#define ADDR 0x40
#define WDATA 0x44
#define WDATA_MASK 0x48
#define WP0 0x50
#define WP1 0x54
#define WP2 0x58
#define WP3 0x5c
#define STA 0x70
#define RDATA0 0x80
#define SHADOW_RDATA0 0xc0
#define SHADOW_RDATA1 0xc4
#define SHADOW_RDATA2 0xc8
#define SHADOW_RDATA3 0xcc
#define SHADOW_RDATA4 0xd0
#define SHADOW_RDATA5 0xd4
#define SHADOW_RDATA6 0xd8
#define SHADOW_RDATA7 0xdc
#define TEE_SYS_EFUSE_LC_PRELD_OFF 0x64
#define TEE_SYS_EFUSE_DBG_KEY1_OFF 0x70
#define ENABLE_DFT_FUNC_MASK GENMASK(3, 0)
#define ENABLE_DFT_FUNC 0x5
#define DISABLE_DFT_FUNC 0xa
/* bit definition for CON */
#define EFUSE_CON_POWER_MSK BIT(14)
/* bit definition for STA */
#define EFUSE_STA_IDLE_MSK BIT(0)
#define EFUSE_STA_RD_STATUS_POS 4
#define EFUSE_STA_RD_STATUS_MSK (0x7UL << EFUSE_STA_RD_STATUS_POS)
#define EFUSE_STA_WR_STATUS_POS 8
#define EFUSE_STA_WR_STATUS_MSK (0x7UL << EFUSE_STA_WR_STATUS_POS)
#define EFUSE_STA_CMD_ILLEGAL_POS 11
#define EFUSE_STA_CMD_ILLEGAL_MSK (0x1UL << EFUSE_STA_CMD_ILLEGAL_POS)
#define EFUSE_STA_KTRANS_ALARM_POS 14
#define EFUSE_STA_KTRANS_ALARM_MSK (0x1UL << EFUSE_STA_KTRANS_ALARM_POS)
/* Max try time for idle check */
#define MAX_TRY_TIME_IDLE 10000
#define DEVICE_BUSY 1
#define EFUSE_CON_CMD_POS 8
#define EFUSE_CON_CMD_MSK GENMASK(12, 8)
#define EFUSE_CON_CMD_IDLE (0x0 << EFUSE_CON_CMD_POS)
#define EFUSE_CON_CMD_READ (0x1 << EFUSE_CON_CMD_POS)
#define EFUSE_CON_CMD_WRITE (0x2 << EFUSE_CON_CMD_POS)
#define EFUSE_CON_CMD_BLKREAD (0x3 << EFUSE_CON_CMD_POS)
#define EFUSE_CON_CMD_WP_LOCK (0x8 << EFUSE_CON_CMD_POS)
#define EFUSE_CON_CMD_CP_LOCK (0x9 << EFUSE_CON_CMD_POS)
#define EFUSE_CON_CMD_RP_LOCK (0xA << EFUSE_CON_CMD_POS)
#define EFUSE_CON_CMD_UP_LC (0x10 << EFUSE_CON_CMD_POS)
#define EFUSE_CON_START BIT(0)
#define EFUSE_CON_CLEAR BIT(1)
#define EFUSE_CON_KEY_TRANS_MSK BIT(13)
#define EFUSE_CON_LC_READ_MSK BIT(2)
#define EFUSE_CON_START_MSK BIT(0)
#define EFUSE_CON_MSK (EFUSE_CON_LC_READ_MSK | \
EFUSE_CON_CMD_MSK | \
EFUSE_CON_KEY_TRANS_MSK | \
EFUSE_CON_START_MSK)
#define IS_CVKEY1(addr) ((addr >= 0x38) && (addr < 0x3C))
#define IS_CVKEY2(addr) ((addr >= 0x3C) && (addr < 0x78))
#define IS_USRKEY2(addr) ((addr >= 0x78) && (addr < 0xd8))
/* Block width (bytes) definitions */
#define LIGHT_EFUSE_LIT_BLOCK_WIDTH 16
#define LIGHT_EFUSE_BIG_BLOCK_WIDTH 32
#define LIGHT_EFUSE_LIT_BLOCK_NUM 52
#define LIGHT_EFUSE_BIG_BLOCK_NUM 6
#define RMA_LIFE_CYCLE_PARA 0x1A946F9B
#define RIP_LIFE_CYCLE_PARA 0xEE45E8A7
struct light_efuse_priv {
struct device *dev;
void __iomem *base;
struct regmap *teesys_regs;
struct clk *clk;
u32 sysfs_rd_offset;
u32 sysfs_rd_len;
};
static u32 perm_spi_magic[] = {
0x9804E1BC,
0x4B8B59F5,
0x36D33417,
0x7491B7D5,
};
static u32 update_lc_magic[] = {
0x768A7E2F,
0xE4D53282,
0x8BD97337,
0x677B9E85,
};
static u32 read_magic[] = {
0x32224E05,
0xC3F981D0,
0xF4D7FB08,
0xA4C8C6DE,
};
static u32 write_magic[] = {
0xB4BC4A0A,
0x2A8B7E6F,
0x974B25A1,
0x67DB5F5F,
};
static u32 block_read_magic[] = {
0x39CF83C1,
0xD0DDD6B2,
0xBD50693B,
0x5F61B752,
};
static u32 wp_lock_magic[] = {
0x0D11ECA6,
0x06EDF631,
0xB58CA544,
0x1EBDE503,
};
static u32 cp_lock_magic[] = {
0xC21E9BB8,
0x0FC428F1,
0xD8E95026,
0x1C34AC41,
};
static u32 rp_lock_magic[] = {
0xAEB3089A,
0x8DE56E9A,
0x453416C2,
0x969F6937,
};
static u32 key_tran_magic[] = {
0x1AF5952C,
0x111B5E55,
0xFAE8A83D,
0xEDFE9E7F,
};
static u32 *cmd_perm_magic_num[] = {
perm_spi_magic,
update_lc_magic,
read_magic,
write_magic,
block_read_magic,
wp_lock_magic,
cp_lock_magic,
rp_lock_magic,
key_tran_magic
};
enum permission_type {
CMD_SPI = 0,
CMD_UPDATE_LC,
CMD_READ,
CMD_WRITE,
CMD_BLOCK_READ,
CMD_WP_LOCK,
CMD_CP_LOCK,
CMD_RP_LOCK,
CMD_KEY_TRAN,
CMD_KEY_MAX,
};
enum con_cmd_type {
CON_CMD_IDLE = 0,
CON_CMD_READ,
CON_CMD_WRITE,
CON_CMD_BLOCK_RD,
CON_CMD_WP_LOCK,
CON_CMD_CP_LOCK,
CON_CMD_RP_LOCK,
CON_CMD_UP_LC,
CON_CMD_MAX,
};
static inline bool efuse_poweron_status_get(void __iomem *base)
{
return readl(base + CON) & EFUSE_CON_POWER_MSK ? false : true;
}
static inline int efuse_idle_check(void __iomem *base)
{
int try_cnt = MAX_TRY_TIME_IDLE;
while (try_cnt--) {
if (!(readl(base + STA) & EFUSE_STA_IDLE_MSK))
return 0;
}
if (try_cnt <= 0)
return -DEVICE_BUSY;
return 0;
}
static inline int efuse_status_check(void __iomem *base)
{
u32 data = readl(base + STA);
int errcode;
errcode = data & (EFUSE_STA_RD_STATUS_MSK | EFUSE_STA_WR_STATUS_MSK |
EFUSE_STA_CMD_ILLEGAL_MSK | EFUSE_STA_KTRANS_ALARM_MSK);
pr_debug("[%s,%d]efuse status before clear: 0x%x\n", __func__, __LINE__, errcode);
/* If error happens, write clear should be added */
if (errcode) {
pr_err("error efuse operation STA status: 0x%x\n", errcode);
writel(data, base + STA);
}
return -errcode;
}
static inline int efuse_poweron(void __iomem *base)
{
u32 data;
int ret;
if (efuse_poweron_status_get(base))
return 0;
data = readl(base + CON);
data &= ~EFUSE_CON_POWER_MSK;
writel(data, base + CON);
ret = efuse_idle_check(base);
ret |= efuse_status_check(base);
pr_debug("pd status: 0x%lx\n", readl(base + CON) & EFUSE_CON_POWER_MSK);
return ret;
}
static inline u32 efuse_data_read(void __iomem *base)
{
return readl(base + RDATA0);
}
static inline void efuse_data_clear(void __iomem *base)
{
u32 data = readl(base + CON);
data |= EFUSE_CON_CLEAR;
writel(data, base + CON);
}
static
inline void efuse_permission_magic_config(void __iomem *base, u32 *magic_num[],
enum permission_type cmd)
{
writel(magic_num[cmd][3], base + WP0);
writel(magic_num[cmd][2], base + WP1);
writel(magic_num[cmd][1], base + WP2);
writel(magic_num[cmd][0], base + WP3);
}
static inline void efuse_addr_config(void __iomem *base, u32 addr)
{
writel(addr, base + ADDR);
pr_debug("[%s, %d]efuse addr reg: 0x%x\n", __func__, __LINE__, readl(base + ADDR));
}
static inline void efuse_data_mask_config(void __iomem *base, u32 mask)
{
writel(mask, base + WDATA_MASK);
}
static inline void efuse_data_config(void __iomem *base, u32 data)
{
writel(data, base + WDATA);
pr_debug("[%s, %d]efuse data reg: 0x%x\n", __func__, __LINE__, readl(base + WDATA));
}
static inline void efuse_life_cycle_para_config(void __iomem *base, u32 data)
{
writel(data, base + LCPAR);
}
static inline u32 efuse_life_cycle_para_get(void __iomem *base)
{
return readl(base + LCPAR);
}
static inline int efuse_cmd_start(void __iomem *base, enum con_cmd_type cmd_type)
{
u32 data, cmd;
switch (cmd_type) {
case CON_CMD_IDLE:
cmd = EFUSE_CON_CMD_IDLE;
break;
case CON_CMD_READ:
cmd = EFUSE_CON_CMD_READ;
break;
case CON_CMD_WRITE:
cmd = EFUSE_CON_CMD_WRITE;
break;
case CON_CMD_BLOCK_RD:
cmd = EFUSE_CON_CMD_BLKREAD;
break;
case CON_CMD_WP_LOCK:
cmd = EFUSE_CON_CMD_WP_LOCK;
break;
case CON_CMD_CP_LOCK:
cmd = EFUSE_CON_CMD_CP_LOCK;
break;
case CON_CMD_RP_LOCK:
cmd = EFUSE_CON_CMD_RP_LOCK;
break;
case CON_CMD_UP_LC:
cmd = EFUSE_CON_CMD_UP_LC;
break;
default:
return -EINVAL;
}
/* Mask LC_Read, Key_transfer, command and start bits */
data = readl(base + CON);
data &= ~EFUSE_CON_MSK;
data |= cmd | EFUSE_CON_START;
writel(data, base + CON);
return 0;
}
static DEFINE_MUTEX(light_efuse_mutex);
static int light_efuse_read_start(void __iomem *base, u32 addr, enum con_cmd_type cmd_type)
{
int ret = 0;
enum permission_type permission;
if (cmd_type == CON_CMD_READ)
permission = CMD_READ;
else if (cmd_type == CON_CMD_BLOCK_RD)
permission = CMD_BLOCK_READ;
else {
pr_err("invaid efuse read command type\n");
return -EINVAL;
}
ret = efuse_idle_check(base);
if (ret) {
pr_err("the device is busy\n");
return ret;
}
efuse_permission_magic_config(base, cmd_perm_magic_num, permission);
efuse_addr_config(base, addr);
ret = efuse_cmd_start(base, cmd_type);
if (ret)
return ret;
/* Wait controller completed */
ret = efuse_idle_check(base);
/* Check status, if there has error, reort and clear status */
ret |= efuse_status_check(base);
if (ret) {
pr_err("error occurs while start reading at efuse byte addr: %d\n", addr * 4);
return ret;
}
if (cmd_type == CON_CMD_BLOCK_RD) {
pr_debug("=======================================\n");
pr_debug("shadow: 0x%x\n", readl(base + SHADOW_RDATA0));
pr_debug("shadow: 0x%x\n", readl(base + SHADOW_RDATA1));
pr_debug("shadow: 0x%x\n", readl(base + SHADOW_RDATA2));
pr_debug("shadow: 0x%x\n", readl(base + SHADOW_RDATA3));
pr_debug("shadow: 0x%x\n", readl(base + SHADOW_RDATA4));
pr_debug("shadow: 0x%x\n", readl(base + SHADOW_RDATA5));
pr_debug("shadow: 0x%x\n", readl(base + SHADOW_RDATA6));
pr_debug("shadow: 0x%x\n", readl(base + SHADOW_RDATA7));
}
return ret;
}
static int light_efuse_read_word(void __iomem *base, u32 addr, u32 *val)
{
int ret = 0;
ret = efuse_idle_check(base);
if (ret)
return ret;
ret = light_efuse_read_start(base, addr, CON_CMD_READ);
if (ret) {
pr_err("failed to start efuse read\n");
goto exit;
}
*val = efuse_data_read(base);
pr_debug("[%s][%d]data = 0x%x\n", __func__, __LINE__,*val);
exit:
efuse_data_clear(base);
return ret;
}
static int light_efuse_write_word(void __iomem *base, u32 addr, u32 data, u32 mask)
{
int ret = 0;
ret = efuse_idle_check(base);
if (ret)
return ret;
/*
* Check permission:
* Check it every time to avoid wp0~3 are changed somewhere
*/
efuse_permission_magic_config(base, cmd_perm_magic_num, CMD_WRITE);
/* Config address */
efuse_addr_config(base, addr);
/* Config data */
efuse_data_config(base, data);
/* Config data mask , if we're in keyrange mask should be set to 0 */
if (IS_CVKEY1(addr) || IS_CVKEY2(addr) || IS_USRKEY2(addr))
efuse_data_mask_config(base, 0);
else
efuse_data_mask_config(base, mask);
/* Set write command */
ret = efuse_cmd_start(base, CON_CMD_WRITE);
if (ret)
goto exit;
/* Wait controller completed */
ret = efuse_idle_check(base);
exit:
/* Check status, if there has error, reort and clear status */
ret |= efuse_status_check(base);
if (ret)
pr_err("error occurs while start writing at efuse byte addr: %d\n", addr * 4);
efuse_data_clear(base);
return ret;
}
static int light_efuse_read(void *context, unsigned int addr, void *data, size_t bytes)
{
struct light_efuse_priv *priv = context;
u32 byte_offset, read_count, read_addr;
u8 *pdst, *psrc;
u32 value;
int ret = 0;
mutex_lock(&light_efuse_mutex);
dev_dbg(priv->dev, "[%s]efuse addr: 0x%x, bytes: %d\n", __func__, addr, (int)bytes);
ret = pm_runtime_get_sync(priv->dev);
if (ret < 0) {
dev_err(priv->dev, "failed to get the efuse device(%d)\n", ret);
pm_runtime_put_noidle(priv->dev);
goto read_end;
}
if (efuse_poweron(priv->base)) {
dev_err(priv->dev, "failed to power on efuse\n");
ret = -EBUSY;
goto read_end;
}
pdst = data;
byte_offset = addr & 0x3;
read_addr = addr / 4; /* Efuse unit is 4 bytes */
/* byte_offset != 0, means not 4 bytes aligned, read first word first */
if (byte_offset) {
ret = light_efuse_read_word(priv->base, read_addr, &value);
if (ret) {
dev_err(priv->dev, "failed to read efuse data\n");
goto read_end;
}
read_count = 4 - byte_offset;
psrc = (u8 *)&value + byte_offset;
if (bytes < read_count)
read_count = bytes;
memcpy(pdst, psrc, read_count);
read_addr++;
pdst += read_count;
bytes -= read_count;
}
while (bytes >= 4) {
ret = light_efuse_read_word(priv->base, read_addr, &value);
if (ret) {
dev_err(priv->dev, "failed to read efuse data\n");
goto read_end;
}
memcpy(pdst, &value, 4);
bytes -= 4;
read_addr++; /* the hardware will span over one word length automatically */
pdst += 4;
}
if (bytes > 0) {
ret = light_efuse_read_word(priv->base, read_addr, &value);
if (ret) {
dev_err(priv->dev, "failed to read data from efuse\n");
goto read_end;
}
memcpy(pdst, &value, bytes);
}
pm_runtime_put_sync(priv->dev);
read_end:
mutex_unlock(&light_efuse_mutex);
return ret;
}
static int light_efuse_write(void *context, unsigned int addr, void *data, size_t bytes)
{
struct light_efuse_priv *priv = context;
int ret = 0;
u32 byte_offset, write_addr, value = 0;
u32 write_count, mask;
u8 *psrc, *pdst;
size_t __maybe_unused orign_bytes = bytes;
mutex_lock(&light_efuse_mutex);
dev_dbg(priv->dev, "[%s]efuse addr: 0x%x, bytes: %d\n", __func__, addr, (int)bytes);
ret = pm_runtime_get_sync(priv->dev);
if (ret < 0) {
dev_err(priv->dev, "failed to get the efuse device(%d)\n", ret);
pm_runtime_put_noidle(priv->dev);
goto write_end;
}
if (efuse_poweron(priv->base)) {
dev_err(priv->dev, "failed to power on efuse\n");
ret = -EBUSY;
goto write_end;
}
byte_offset = addr & 0x3;
psrc = (u8 *)data;
write_addr = addr / 4;
pr_debug("[%s][%d]: write addr = 0x%x\n", __func__, __LINE__, write_addr);
pr_debug("Write data: ");
if (byte_offset) {
write_count = 4 - byte_offset;
if (bytes < write_count)
write_count = bytes;
pdst = (u8 *)&value + byte_offset;
memcpy(pdst, psrc, write_count);
pr_debug("0x%x ", *psrc);
mask = ~value;
ret = light_efuse_write_word(priv->base, write_addr, value, mask);
if (ret) {
dev_err(priv->dev, "failed to write data to efuse\n");
goto write_end;
}
psrc += write_count;
write_addr++;
bytes -= write_count;
}
while (bytes >= 4) {
value = 0;
pdst = (u8 *)&value;
write_count = 4;
memcpy(pdst, psrc, write_count);
pr_debug("0x%x ", *psrc);
mask = ~value;
ret = light_efuse_write_word(priv->base, write_addr, value, mask);
if (ret) {
dev_err(priv->dev, "failed to write data to efuse\n");
goto write_end;
}
psrc += write_count;
bytes -= write_count;
write_addr++;
}
if (bytes > 0) {
value = 0;
pdst = (u8 *)&value;
memcpy(pdst, psrc, bytes);
pr_debug("0x%x ", *psrc);
mask = ~value;
ret = light_efuse_write_word(priv->base, write_addr, value, mask);
if (ret) {
dev_err(priv->dev, "failed to write data to efuse\n");
goto write_end;
}
}
pr_debug("\n");
pm_runtime_put_sync(priv->dev);
write_end:
mutex_unlock(&light_efuse_mutex);
return ret < 0 ? ret : orign_bytes;
}
static ssize_t efuse_nvmem_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t count)
{
struct light_efuse_priv *priv = dev_get_drvdata(dev);
char *start = (char *)buf;
char type;
unsigned long addr, len;
unsigned char *data;
int i, ret;
/*
* echo types:
* echo w offset len 0x01 0x02 0x03 ... > efuse_nvmem
* echo r offset len > efuse_nvmem
*/
while (*start == ' ') /* skip space */
start++;
if (*start != 'w' && *start != 'r')
return -EINVAL;
type = *start;
start++;
while (*start == ' ')
start++;
addr = simple_strtoul(start, &start, 0);
while (*start == ' ')
start++;
len = simple_strtoul(start, &start, 0);
priv->sysfs_rd_offset = addr;
priv->sysfs_rd_len = len;
if (type == 'r')
goto exit;
data = kzalloc(sizeof(*data) * len, GFP_KERNEL);
if (!data)
return -ENOMEM;
pr_debug("echo data:\n");
for (i = 0; i < len; i++) {
while (*start == ' ')
start++;
data[i] = simple_strtoul(start, &start, 0);
pr_debug("0x%x ", data[i]);
}
ret = light_efuse_write(priv, addr, data, len);
if (ret < 0) {
kfree(data);
return ret;
}
kfree(data);
exit:
return count;
}
static ssize_t efuse_nvmem_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct light_efuse_priv *priv = dev_get_drvdata(dev);
u32 addr = priv->sysfs_rd_offset;
u32 len = priv->sysfs_rd_len;
int ret, i;
unsigned char *data;
size_t bufpos = 0, count;
data = kzalloc(sizeof(*data) * len, GFP_KERNEL);
if (!data)
return -ENOMEM;
ret = light_efuse_read(priv, addr, data, len);
if (ret < 0)
goto out;
count = (len + 2) * 10;
for (i = 0; i < len; i++) {
snprintf(buf + bufpos, count - bufpos, "0x%.*x", 2, data[i]);
bufpos += 4;
if (i == len - 1 || (i !=0 && i % 16 == 0))
buf[bufpos++] = '\n';
else
buf[bufpos++] = ' ';
}
out:
kfree(data);
return bufpos;
}
static ssize_t rma_lc_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t count)
{
struct light_efuse_priv *priv = dev_get_drvdata(dev);
u32 value = RMA_LIFE_CYCLE_PARA;
int ret;
ret = pm_runtime_get_sync(priv->dev);
if (ret < 0) {
dev_err(priv->dev, "failed to get the efuse device(%d)\n", ret);
pm_runtime_put_noidle(priv->dev);
return ret;
}
regmap_update_bits(priv->teesys_regs,
TEE_SYS_EFUSE_DBG_KEY1_OFF,
ENABLE_DFT_FUNC_MASK,
ENABLE_DFT_FUNC);
efuse_permission_magic_config(priv->base, cmd_perm_magic_num, CMD_UPDATE_LC);
efuse_life_cycle_para_config(priv->base, value);
/* Set command */
ret = efuse_cmd_start(priv->base, CON_CMD_UP_LC);
if (ret)
goto exit;
/* Wait controller completed */
ret = efuse_idle_check(priv->base);
pr_debug("set life cycle value: 0x%x\n", value);
exit:
regmap_update_bits(priv->teesys_regs,
TEE_SYS_EFUSE_DBG_KEY1_OFF,
ENABLE_DFT_FUNC_MASK,
DISABLE_DFT_FUNC);
/* Check status, if there has error, reort and clear status */
ret |= efuse_status_check(priv->base);
if (ret)
pr_err("error occurs while starting write\n");
efuse_data_clear(priv->base);
pm_runtime_put_sync(priv->dev);
return ret < 0 ? ret : count;
}
static ssize_t rip_lc_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t count)
{
struct light_efuse_priv *priv = dev_get_drvdata(dev);
u32 value = RIP_LIFE_CYCLE_PARA;
int ret;
ret = pm_runtime_get_sync(priv->dev);
if (ret < 0) {
dev_err(priv->dev, "failed to get the efuse device(%d)\n", ret);
pm_runtime_put_noidle(priv->dev);
return ret;
}
efuse_permission_magic_config(priv->base, cmd_perm_magic_num, CMD_UPDATE_LC);
efuse_life_cycle_para_config(priv->base, value);
/* Set command */
ret = efuse_cmd_start(priv->base, CON_CMD_UP_LC);
if (ret)
goto exit;
/* Wait controller completed */
ret = efuse_idle_check(priv->base);
exit:
/* Check status, if there has error, reort and clear status */
ret |= efuse_status_check(priv->base);
if (ret)
pr_err("error occurs while starting write\n");
efuse_data_clear(priv->base);
pm_runtime_put_sync(priv->dev);
return ret < 0 ? ret : count;
}
static ssize_t lc_preld_show(struct device *dev,
struct device_attribute *attr, char *buf)
{
struct light_efuse_priv *priv = dev_get_drvdata(dev);
int ret;
u32 data;
ret = regmap_read(priv->teesys_regs, TEE_SYS_EFUSE_LC_PRELD_OFF, &data);
if (ret) {
dev_err(dev, "failed to read data from LC_PRELD area\n");
return ret;
}
return sprintf(buf, "0x%08x\n", data);
}
static ssize_t update_lc_store(struct device *dev,
struct device_attribute *attr,
const char *buf, size_t count)
{
struct light_efuse_priv *priv = dev_get_drvdata(dev);
int ret;
u32 value, data;
const char *p, *life_cycle = buf;
int len;
p = memchr(buf, '\n', count);
len = p ? p - buf : count;
dev_dbg(dev, "life_cycle: %s, buf: %s, len: %d\n", life_cycle, buf, len);
if (!strncmp(life_cycle, "LC_RMA", len)) {
/* If target life cycle is RMA, open permission in teesystem regs */
ret = regmap_read(priv->teesys_regs,
TEE_SYS_EFUSE_DBG_KEY1_OFF,
&data); /* Register from tee system */
if (ret) {
dev_err(dev, "failed to read data from DBG_KEY1 area\n");
return ret;
}
data &= ~0xf;
data |= 0x5;
ret = regmap_write(priv->teesys_regs,
TEE_SYS_EFUSE_DBG_KEY1_OFF,
data);
if (ret) {
dev_err(dev, "failed to write data to DBG_KEY1 area\n");
return ret;
}
value = 0x1A946F9B;
} else if (!strncmp(life_cycle, "LC_OEM", len))
value = 0x64EA9B8E;
else if (!strncmp(life_cycle, "LC_PRO", len))
value = 0xB0E047A8;
else if (!strncmp(life_cycle, "LC_DEV", len))
value = 0x59DD3BDF;
else if (!strncmp(life_cycle, "LC_RIP", len))
value = 0xEE45E8A7;
else if (!strncmp(life_cycle, "LC_KILL_KEY1", len))
value = 0x7D8E9CA1;
else if (!strncmp(life_cycle, "LC_KILL_KEY0", len))
value = 0xC29F604B;
else {
dev_err(dev, "invalid life cycle type!\n");
return -EINVAL;
}
/*
* Check permission:
* Check it every time to avoid wp0~3 are changed somewhere
*/
efuse_permission_magic_config(priv->base, cmd_perm_magic_num, CMD_UPDATE_LC);
/* Config life cycle */
efuse_life_cycle_para_config(priv->base, value);
/* set command */
ret = efuse_cmd_start(priv->base, CON_CMD_UP_LC);
if (ret)
goto exit;
/* Wait controller completed */
ret = efuse_idle_check(priv->base);
exit:
/* Check status, if there has error, reort and clear status */
ret |= efuse_status_check(priv->base);
if (ret)
dev_err(dev, "error occurs while starting write\n");
efuse_data_clear(priv->base);
if (strncmp(life_cycle, "LC_RMA", len))
goto out;
dev_info(dev, "set LC_RMA life cycle\n");
/* If target life cycle is RMA, close permission in teesystem regs */
ret = regmap_read(priv->teesys_regs,
TEE_SYS_EFUSE_DBG_KEY1_OFF,
&data); /* Register from tee system */
if (ret) {
dev_err(dev, "failed to read data from DBG_KEY1 area\n");
return ret;
}
data &= ~0xf;
data |= 0xa;
ret = regmap_write(priv->teesys_regs,
TEE_SYS_EFUSE_DBG_KEY1_OFF,
data);
if (ret) {
dev_err(dev, "failed to write data to DBG_KEY1 area\n");
return ret;
}
out:
return ret < 0 ? ret : count;
}
static DEVICE_ATTR_WO(rma_lc);
static DEVICE_ATTR_WO(rip_lc);
static DEVICE_ATTR_RW(efuse_nvmem);
static DEVICE_ATTR_RO(lc_preld);
static DEVICE_ATTR_WO(update_lc);
static struct attribute *light_efuse_sysfs_entries[] = {
&dev_attr_efuse_nvmem.attr,
&dev_attr_rip_lc.attr,
&dev_attr_rma_lc.attr,
&dev_attr_lc_preld.attr,
&dev_attr_update_lc.attr,
NULL
};
static const struct attribute_group dev_attr_efuse_sysfs_group = {
.attrs = light_efuse_sysfs_entries,
};
static const struct of_device_id light_efuse_of_match[] = {
{.compatible = "thead,light-fm-efuse"},
{ /* sentinel */},
};
MODULE_DEVICE_TABLE(of, light_efuse_of_match);
static int __maybe_unused light_efuse_runtime_suspend(struct device *dev)
{
struct light_efuse_priv *priv = dev_get_drvdata(dev);
u32 data;
int ret;
dev_dbg(dev, "[%s,%d]efuse runtime power down\n", __func__, __LINE__);
if (!efuse_poweron_status_get(priv->base))
return 0;
data = readl(priv->base + CON);
data |= EFUSE_CON_POWER_MSK;
writel(data, priv->base + CON);
ret = efuse_idle_check(priv->base);
ret |= efuse_status_check(priv->base);
dev_dbg(dev, "[%s,%d] ret = %d, pd status: 0x%lx\n", __func__, __LINE__, ret,
readl(priv->base + CON) & EFUSE_CON_POWER_MSK);
clk_disable_unprepare(priv->clk);
return ret;
}
static int __maybe_unused light_efuse_runtime_resume(struct device *dev)
{
struct light_efuse_priv *priv = dev_get_drvdata(dev);
int ret;
dev_dbg(dev, "[%s,%d]efuse runtime power on\n", __func__, __LINE__);
ret = clk_prepare_enable(priv->clk);
if (ret) {
dev_err(dev, "failed to get efuse clk\n");
return ret;
}
return efuse_poweron(priv->base);
}
static int __maybe_unused light_efuse_suspend(struct device *dev)
{
struct light_efuse_priv *priv = dev_get_drvdata(dev);
u32 data;
int ret;
dev_dbg(dev, "[%s,%d]efuse suspend enter\n", __func__, __LINE__);
if (!efuse_poweron_status_get(priv->base))
return 0;
data = readl(priv->base + CON);
data |= EFUSE_CON_POWER_MSK;
writel(data, priv->base + CON);
ret = efuse_idle_check(priv->base);
ret |= efuse_status_check(priv->base);
dev_dbg(dev, "[%s,%d] ret = %d, pd status: 0x%lx\n", __func__, __LINE__, ret,
readl(priv->base + CON) & EFUSE_CON_POWER_MSK);
clk_disable_unprepare(priv->clk);
return ret;
}
static int __maybe_unused light_efuse_resume(struct device *dev)
{
struct light_efuse_priv *priv = dev_get_drvdata(dev);
int ret;
dev_dbg(dev, "[%s,%d]efuse resume enter\n", __func__, __LINE__);
ret = clk_prepare_enable(priv->clk);
if (ret) {
dev_err(dev, "failed to get efuse clk\n");
return ret;
}
return efuse_poweron(priv->base);
}
static int light_efuse_probe(struct platform_device *pdev)
{
struct device *dev = &pdev->dev;
struct nvmem_device *nvmem;
struct nvmem_config config = {};
struct light_efuse_priv *priv;
int ret;
priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
if (!priv)
return -ENOMEM;
priv->base = devm_platform_ioremap_resource(pdev, 0);
if (IS_ERR(priv->base))
return PTR_ERR(priv->base);
/* optional clock, default open */
priv->clk = devm_clk_get(dev, "pclk");
if (IS_ERR_OR_NULL(priv->clk)) {
if (PTR_ERR(priv->clk) != -EPROBE_DEFER)
dev_err(&pdev->dev, "failed to get efuse clk\n");
return PTR_ERR(priv->clk);
}
ret = clk_prepare_enable(priv->clk);
if (ret) {
dev_err(dev, "failed to get efuse clk\n");
return ret;
}
priv->teesys_regs = syscon_regmap_lookup_by_phandle(dev->of_node, "thead,teesys");
if (IS_ERR(priv->teesys_regs)) {
dev_err(dev, "unable to find teesys registers\n");
return PTR_ERR(priv->teesys_regs);
}
priv->dev = dev;
dev_set_drvdata(dev, priv);
pm_runtime_enable(dev);
ret = pm_runtime_get_sync(dev);
if (ret < 0) {
dev_err(dev, "failed to get the efuse device(%d)\n", ret);
pm_runtime_put_noidle(dev);
return ret;
}
ret = sysfs_create_group(&dev->kobj, &dev_attr_efuse_sysfs_group);
if (ret) {
dev_err(dev, "failed to create efuse debug sysfs\n");
return ret;
}
config.name = "light-efuse";
config.read_only = false;
config.stride = 1;
config.word_size = 1; /* the least read and write unit on the upper level */
config.reg_read = light_efuse_read;
config.reg_write = light_efuse_write;
config.size = LIGHT_EFUSE_LIT_BLOCK_NUM * LIGHT_EFUSE_LIT_BLOCK_WIDTH +
LIGHT_EFUSE_BIG_BLOCK_NUM * LIGHT_EFUSE_BIG_BLOCK_WIDTH;
config.priv = priv;
config.dev = dev;
nvmem = devm_nvmem_register(dev, &config);
if (IS_ERR(nvmem))
return PTR_ERR_OR_ZERO(nvmem);
pm_runtime_put_sync(dev);
dev_info(dev, "succeed to register light efuse driver\n");
return 0;
}
static const struct dev_pm_ops efuse_runtime_pm_ops = {
SET_RUNTIME_PM_OPS(light_efuse_runtime_suspend, light_efuse_runtime_resume, NULL)
SET_SYSTEM_SLEEP_PM_OPS(light_efuse_suspend, light_efuse_resume)
};
static struct platform_driver light_efuse_driver = {
.probe = light_efuse_probe,
.driver = {
.name = "light_efuse",
.of_match_table = light_efuse_of_match,
.pm = &efuse_runtime_pm_ops,
},
};
module_platform_driver(light_efuse_driver);
MODULE_AUTHOR("wei.liu <lw312886@linux.alibaba.com>");
MODULE_DESCRIPTION("Thead light nvmem driver");
MODULE_LICENSE("GPL v2");