mirror of
https://github.com/revyos/thead-kernel.git
synced 2026-06-21 09:12:26 +02:00
1167 lines
27 KiB
C
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");
|