mirror of
https://github.com/AsahiLinux/u-boot
synced 2024-11-14 08:57:58 +00:00
b5392c50bd
Add driver for the efuse block in the JZ47xx SOC. Cc: Daniel Schwierzeck <daniel.schwierzeck@gmail.com> Signed-off-by: Paul Burton <paul.burton@imgtec.com> Signed-off-by: Marek Vasut <marek.vasut@gmail.com> Signed-off-by: Ezequiel Garcia <ezequiel@collabora.com> Reviewed-by: Marek Vasut <marex@denx.de>
103 lines
2.4 KiB
C
103 lines
2.4 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* JZ4780 EFUSE driver
|
|
*
|
|
* Copyright (c) 2014 Imagination Technologies
|
|
* Author: Alex Smith <alex.smith@imgtec.com>
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <asm/io.h>
|
|
#include <asm/unaligned.h>
|
|
#include <errno.h>
|
|
#include <mach/jz4780.h>
|
|
#include <wait_bit.h>
|
|
|
|
#define EFUSE_EFUCTRL 0xd0
|
|
#define EFUSE_EFUCFG 0xd4
|
|
#define EFUSE_EFUSTATE 0xd8
|
|
#define EFUSE_EFUDATA(n) (0xdc + ((n) * 4))
|
|
|
|
#define EFUSE_EFUCTRL_RD_EN BIT(0)
|
|
#define EFUSE_EFUCTRL_LEN_BIT 16
|
|
#define EFUSE_EFUCTRL_LEN_MASK 0x1f
|
|
#define EFUSE_EFUCTRL_ADDR_BIT 21
|
|
#define EFUSE_EFUCTRL_ADDR_MASK 0x1ff
|
|
#define EFUSE_EFUCTRL_CS BIT(30)
|
|
|
|
#define EFUSE_EFUCFG_RD_STROBE_BIT 16
|
|
#define EFUSE_EFUCFG_RD_STROBE_MASK 0xf
|
|
#define EFUSE_EFUCFG_RD_ADJ_BIT 20
|
|
#define EFUSE_EFUCFG_RD_ADJ_MASK 0xf
|
|
|
|
#define EFUSE_EFUSTATE_RD_DONE BIT(0)
|
|
|
|
static void jz4780_efuse_read_chunk(size_t addr, size_t count, u8 *buf)
|
|
{
|
|
void __iomem *regs = (void __iomem *)NEMC_BASE;
|
|
size_t i;
|
|
u32 val;
|
|
int ret;
|
|
|
|
val = EFUSE_EFUCTRL_RD_EN |
|
|
((count - 1) << EFUSE_EFUCTRL_LEN_BIT) |
|
|
(addr << EFUSE_EFUCTRL_ADDR_BIT) |
|
|
((addr > 0x200) ? EFUSE_EFUCTRL_CS : 0);
|
|
writel(val, regs + EFUSE_EFUCTRL);
|
|
|
|
ret = wait_for_bit_le32(regs + EFUSE_EFUSTATE,
|
|
EFUSE_EFUSTATE_RD_DONE, true, 10000, false);
|
|
if (ret)
|
|
return;
|
|
|
|
if ((count % 4) == 0) {
|
|
for (i = 0; i < count / 4; i++) {
|
|
val = readl(regs + EFUSE_EFUDATA(i));
|
|
put_unaligned(val, (u32 *)(buf + (i * 4)));
|
|
}
|
|
} else {
|
|
val = readl(regs + EFUSE_EFUDATA(0));
|
|
if (count > 2)
|
|
buf[2] = (val >> 16) & 0xff;
|
|
if (count > 1)
|
|
buf[1] = (val >> 8) & 0xff;
|
|
buf[0] = val & 0xff;
|
|
}
|
|
}
|
|
|
|
static inline int jz4780_efuse_chunk_size(size_t count)
|
|
{
|
|
if (count >= 32)
|
|
return 32;
|
|
else if ((count / 4) > 0)
|
|
return (count / 4) * 4;
|
|
else
|
|
return count % 4;
|
|
}
|
|
|
|
void jz4780_efuse_read(size_t addr, size_t count, u8 *buf)
|
|
{
|
|
size_t chunk;
|
|
|
|
while (count > 0) {
|
|
chunk = jz4780_efuse_chunk_size(count);
|
|
jz4780_efuse_read_chunk(addr, chunk, buf);
|
|
addr += chunk;
|
|
buf += chunk;
|
|
count -= chunk;
|
|
}
|
|
}
|
|
|
|
void jz4780_efuse_init(u32 ahb2_rate)
|
|
{
|
|
void __iomem *regs = (void __iomem *)NEMC_BASE;
|
|
u32 rd_adj, rd_strobe, tmp;
|
|
|
|
rd_adj = (((6500 * (ahb2_rate / 1000000)) / 1000000) + 0xf) / 2;
|
|
tmp = (((35000 * (ahb2_rate / 1000000)) / 1000000) - 4) - rd_adj;
|
|
rd_strobe = ((tmp + 0xf) / 2 < 7) ? 7 : (tmp + 0xf) / 2;
|
|
|
|
tmp = (rd_adj << EFUSE_EFUCFG_RD_ADJ_BIT) |
|
|
(rd_strobe << EFUSE_EFUCFG_RD_STROBE_BIT);
|
|
writel(tmp, regs + EFUSE_EFUCFG);
|
|
}
|