mirror of
https://github.com/AsahiLinux/u-boot
synced 2025-01-15 14:44:02 +00:00
83d290c56f
When U-Boot started using SPDX tags we were among the early adopters and there weren't a lot of other examples to borrow from. So we picked the area of the file that usually had a full license text and replaced it with an appropriate SPDX-License-Identifier: entry. Since then, the Linux Kernel has adopted SPDX tags and they place it as the very first line in a file (except where shebangs are used, then it's second line) and with slightly different comment styles than us. In part due to community overlap, in part due to better tag visibility and in part for other minor reasons, switch over to that style. This commit changes all instances where we have a single declared license in the tag as both the before and after are identical in tag contents. There's also a few places where I found we did not have a tag and have introduced one. Signed-off-by: Tom Rini <trini@konsulko.com>
568 lines
13 KiB
C
568 lines
13 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* S5PC100 OneNAND driver at U-Boot
|
|
*
|
|
* Copyright (C) 2008-2009 Samsung Electronics
|
|
* Kyungmin Park <kyungmin.park@samsung.com>
|
|
*
|
|
* Implementation:
|
|
* Emulate the pseudo BufferRAM
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <malloc.h>
|
|
#include <linux/compat.h>
|
|
#include <linux/mtd/mtd.h>
|
|
#include <linux/mtd/onenand.h>
|
|
#include <linux/mtd/flashchip.h>
|
|
#include <linux/mtd/samsung_onenand.h>
|
|
|
|
#include <asm/io.h>
|
|
#include <linux/errno.h>
|
|
|
|
#define ONENAND_ERASE_STATUS 0x00
|
|
#define ONENAND_MULTI_ERASE_SET 0x01
|
|
#define ONENAND_ERASE_START 0x03
|
|
#define ONENAND_UNLOCK_START 0x08
|
|
#define ONENAND_UNLOCK_END 0x09
|
|
#define ONENAND_LOCK_START 0x0A
|
|
#define ONENAND_LOCK_END 0x0B
|
|
#define ONENAND_LOCK_TIGHT_START 0x0C
|
|
#define ONENAND_LOCK_TIGHT_END 0x0D
|
|
#define ONENAND_UNLOCK_ALL 0x0E
|
|
#define ONENAND_OTP_ACCESS 0x12
|
|
#define ONENAND_SPARE_ACCESS_ONLY 0x13
|
|
#define ONENAND_MAIN_ACCESS_ONLY 0x14
|
|
#define ONENAND_ERASE_VERIFY 0x15
|
|
#define ONENAND_MAIN_SPARE_ACCESS 0x16
|
|
#define ONENAND_PIPELINE_READ 0x4000
|
|
|
|
#if defined(CONFIG_S5P)
|
|
#define MAP_00 (0x0 << 26)
|
|
#define MAP_01 (0x1 << 26)
|
|
#define MAP_10 (0x2 << 26)
|
|
#define MAP_11 (0x3 << 26)
|
|
#endif
|
|
|
|
/* read/write of XIP buffer */
|
|
#define CMD_MAP_00(mem_addr) (MAP_00 | ((mem_addr) << 1))
|
|
/* read/write to the memory device */
|
|
#define CMD_MAP_01(mem_addr) (MAP_01 | (mem_addr))
|
|
/* control special functions of the memory device */
|
|
#define CMD_MAP_10(mem_addr) (MAP_10 | (mem_addr))
|
|
/* direct interface(direct access) with the memory device */
|
|
#define CMD_MAP_11(mem_addr) (MAP_11 | ((mem_addr) << 2))
|
|
|
|
struct s3c_onenand {
|
|
struct mtd_info *mtd;
|
|
void __iomem *base;
|
|
void __iomem *ahb_addr;
|
|
int bootram_command;
|
|
void __iomem *page_buf;
|
|
void __iomem *oob_buf;
|
|
unsigned int (*mem_addr)(int fba, int fpa, int fsa);
|
|
struct samsung_onenand *reg;
|
|
};
|
|
|
|
static struct s3c_onenand *onenand;
|
|
|
|
static int s3c_read_cmd(unsigned int cmd)
|
|
{
|
|
return readl(onenand->ahb_addr + cmd);
|
|
}
|
|
|
|
static void s3c_write_cmd(int value, unsigned int cmd)
|
|
{
|
|
writel(value, onenand->ahb_addr + cmd);
|
|
}
|
|
|
|
/*
|
|
* MEM_ADDR
|
|
*
|
|
* fba: flash block address
|
|
* fpa: flash page address
|
|
* fsa: flash sector address
|
|
*
|
|
* return the buffer address on the memory device
|
|
* It will be combined with CMD_MAP_XX
|
|
*/
|
|
#if defined(CONFIG_S5P)
|
|
static unsigned int s3c_mem_addr(int fba, int fpa, int fsa)
|
|
{
|
|
return (fba << 13) | (fpa << 7) | (fsa << 5);
|
|
}
|
|
#endif
|
|
|
|
static void s3c_onenand_reset(void)
|
|
{
|
|
unsigned long timeout = 0x10000;
|
|
int stat;
|
|
|
|
writel(ONENAND_MEM_RESET_COLD, &onenand->reg->mem_reset);
|
|
while (timeout--) {
|
|
stat = readl(&onenand->reg->int_err_stat);
|
|
if (stat & RST_CMP)
|
|
break;
|
|
}
|
|
stat = readl(&onenand->reg->int_err_stat);
|
|
writel(stat, &onenand->reg->int_err_ack);
|
|
|
|
/* Clear interrupt */
|
|
writel(0x0, &onenand->reg->int_err_ack);
|
|
/* Clear the ECC status */
|
|
writel(0x0, &onenand->reg->ecc_err_stat);
|
|
}
|
|
|
|
static unsigned short s3c_onenand_readw(void __iomem *addr)
|
|
{
|
|
struct onenand_chip *this = onenand->mtd->priv;
|
|
int reg = addr - this->base;
|
|
int word_addr = reg >> 1;
|
|
int value;
|
|
|
|
/* It's used for probing time */
|
|
switch (reg) {
|
|
case ONENAND_REG_MANUFACTURER_ID:
|
|
return readl(&onenand->reg->manufact_id);
|
|
case ONENAND_REG_DEVICE_ID:
|
|
return readl(&onenand->reg->device_id);
|
|
case ONENAND_REG_VERSION_ID:
|
|
return readl(&onenand->reg->flash_ver_id);
|
|
case ONENAND_REG_DATA_BUFFER_SIZE:
|
|
return readl(&onenand->reg->data_buf_size);
|
|
case ONENAND_REG_TECHNOLOGY:
|
|
return readl(&onenand->reg->tech);
|
|
case ONENAND_REG_SYS_CFG1:
|
|
return readl(&onenand->reg->mem_cfg);
|
|
|
|
/* Used at unlock all status */
|
|
case ONENAND_REG_CTRL_STATUS:
|
|
return 0;
|
|
|
|
case ONENAND_REG_WP_STATUS:
|
|
return ONENAND_WP_US;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
/* BootRAM access control */
|
|
if (reg < ONENAND_DATARAM && onenand->bootram_command) {
|
|
if (word_addr == 0)
|
|
return readl(&onenand->reg->manufact_id);
|
|
if (word_addr == 1)
|
|
return readl(&onenand->reg->device_id);
|
|
if (word_addr == 2)
|
|
return readl(&onenand->reg->flash_ver_id);
|
|
}
|
|
|
|
value = s3c_read_cmd(CMD_MAP_11(word_addr)) & 0xffff;
|
|
printk(KERN_INFO "s3c_onenand_readw: Illegal access"
|
|
" at reg 0x%x, value 0x%x\n", word_addr, value);
|
|
return value;
|
|
}
|
|
|
|
static void s3c_onenand_writew(unsigned short value, void __iomem *addr)
|
|
{
|
|
struct onenand_chip *this = onenand->mtd->priv;
|
|
int reg = addr - this->base;
|
|
int word_addr = reg >> 1;
|
|
|
|
/* It's used for probing time */
|
|
switch (reg) {
|
|
case ONENAND_REG_SYS_CFG1:
|
|
writel(value, &onenand->reg->mem_cfg);
|
|
return;
|
|
|
|
case ONENAND_REG_START_ADDRESS1:
|
|
case ONENAND_REG_START_ADDRESS2:
|
|
return;
|
|
|
|
/* Lock/lock-tight/unlock/unlock_all */
|
|
case ONENAND_REG_START_BLOCK_ADDRESS:
|
|
return;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
/* BootRAM access control */
|
|
if (reg < ONENAND_DATARAM) {
|
|
if (value == ONENAND_CMD_READID) {
|
|
onenand->bootram_command = 1;
|
|
return;
|
|
}
|
|
if (value == ONENAND_CMD_RESET) {
|
|
writel(ONENAND_MEM_RESET_COLD,
|
|
&onenand->reg->mem_reset);
|
|
onenand->bootram_command = 0;
|
|
return;
|
|
}
|
|
}
|
|
|
|
printk(KERN_INFO "s3c_onenand_writew: Illegal access"
|
|
" at reg 0x%x, value 0x%x\n", word_addr, value);
|
|
|
|
s3c_write_cmd(value, CMD_MAP_11(word_addr));
|
|
}
|
|
|
|
static int s3c_onenand_wait(struct mtd_info *mtd, int state)
|
|
{
|
|
unsigned int flags = INT_ACT;
|
|
unsigned int stat, ecc;
|
|
unsigned long timeout = 0x100000;
|
|
|
|
switch (state) {
|
|
case FL_READING:
|
|
flags |= BLK_RW_CMP | LOAD_CMP;
|
|
break;
|
|
case FL_WRITING:
|
|
flags |= BLK_RW_CMP | PGM_CMP;
|
|
break;
|
|
case FL_ERASING:
|
|
flags |= BLK_RW_CMP | ERS_CMP;
|
|
break;
|
|
case FL_LOCKING:
|
|
flags |= BLK_RW_CMP;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
while (timeout--) {
|
|
stat = readl(&onenand->reg->int_err_stat);
|
|
if (stat & flags)
|
|
break;
|
|
}
|
|
|
|
/* To get correct interrupt status in timeout case */
|
|
stat = readl(&onenand->reg->int_err_stat);
|
|
writel(stat, &onenand->reg->int_err_ack);
|
|
|
|
/*
|
|
* In the Spec. it checks the controller status first
|
|
* However if you get the correct information in case of
|
|
* power off recovery (POR) test, it should read ECC status first
|
|
*/
|
|
if (stat & LOAD_CMP) {
|
|
ecc = readl(&onenand->reg->ecc_err_stat);
|
|
if (ecc & ONENAND_ECC_4BIT_UNCORRECTABLE) {
|
|
printk(KERN_INFO "%s: ECC error = 0x%04x\n",
|
|
__func__, ecc);
|
|
mtd->ecc_stats.failed++;
|
|
return -EBADMSG;
|
|
}
|
|
}
|
|
|
|
if (stat & (LOCKED_BLK | ERS_FAIL | PGM_FAIL | LD_FAIL_ECC_ERR)) {
|
|
printk(KERN_INFO "%s: controller error = 0x%04x\n",
|
|
__func__, stat);
|
|
if (stat & LOCKED_BLK)
|
|
printk(KERN_INFO "%s: it's locked error = 0x%04x\n",
|
|
__func__, stat);
|
|
|
|
return -EIO;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int s3c_onenand_command(struct mtd_info *mtd, int cmd,
|
|
loff_t addr, size_t len)
|
|
{
|
|
struct onenand_chip *this = mtd->priv;
|
|
unsigned int *m, *s;
|
|
int fba, fpa, fsa = 0;
|
|
unsigned int mem_addr;
|
|
int i, mcount, scount;
|
|
int index;
|
|
|
|
fba = (int) (addr >> this->erase_shift);
|
|
fpa = (int) (addr >> this->page_shift);
|
|
fpa &= this->page_mask;
|
|
|
|
mem_addr = onenand->mem_addr(fba, fpa, fsa);
|
|
|
|
switch (cmd) {
|
|
case ONENAND_CMD_READ:
|
|
case ONENAND_CMD_READOOB:
|
|
case ONENAND_CMD_BUFFERRAM:
|
|
ONENAND_SET_NEXT_BUFFERRAM(this);
|
|
default:
|
|
break;
|
|
}
|
|
|
|
index = ONENAND_CURRENT_BUFFERRAM(this);
|
|
|
|
/*
|
|
* Emulate Two BufferRAMs and access with 4 bytes pointer
|
|
*/
|
|
m = (unsigned int *) onenand->page_buf;
|
|
s = (unsigned int *) onenand->oob_buf;
|
|
|
|
if (index) {
|
|
m += (this->writesize >> 2);
|
|
s += (mtd->oobsize >> 2);
|
|
}
|
|
|
|
mcount = mtd->writesize >> 2;
|
|
scount = mtd->oobsize >> 2;
|
|
|
|
switch (cmd) {
|
|
case ONENAND_CMD_READ:
|
|
/* Main */
|
|
for (i = 0; i < mcount; i++)
|
|
*m++ = s3c_read_cmd(CMD_MAP_01(mem_addr));
|
|
return 0;
|
|
|
|
case ONENAND_CMD_READOOB:
|
|
writel(TSRF, &onenand->reg->trans_spare);
|
|
/* Main */
|
|
for (i = 0; i < mcount; i++)
|
|
*m++ = s3c_read_cmd(CMD_MAP_01(mem_addr));
|
|
|
|
/* Spare */
|
|
for (i = 0; i < scount; i++)
|
|
*s++ = s3c_read_cmd(CMD_MAP_01(mem_addr));
|
|
|
|
writel(0, &onenand->reg->trans_spare);
|
|
return 0;
|
|
|
|
case ONENAND_CMD_PROG:
|
|
/* Main */
|
|
for (i = 0; i < mcount; i++)
|
|
s3c_write_cmd(*m++, CMD_MAP_01(mem_addr));
|
|
return 0;
|
|
|
|
case ONENAND_CMD_PROGOOB:
|
|
writel(TSRF, &onenand->reg->trans_spare);
|
|
|
|
/* Main - dummy write */
|
|
for (i = 0; i < mcount; i++)
|
|
s3c_write_cmd(0xffffffff, CMD_MAP_01(mem_addr));
|
|
|
|
/* Spare */
|
|
for (i = 0; i < scount; i++)
|
|
s3c_write_cmd(*s++, CMD_MAP_01(mem_addr));
|
|
|
|
writel(0, &onenand->reg->trans_spare);
|
|
return 0;
|
|
|
|
case ONENAND_CMD_UNLOCK_ALL:
|
|
s3c_write_cmd(ONENAND_UNLOCK_ALL, CMD_MAP_10(mem_addr));
|
|
return 0;
|
|
|
|
case ONENAND_CMD_ERASE:
|
|
s3c_write_cmd(ONENAND_ERASE_START, CMD_MAP_10(mem_addr));
|
|
return 0;
|
|
|
|
case ONENAND_CMD_MULTIBLOCK_ERASE:
|
|
s3c_write_cmd(ONENAND_MULTI_ERASE_SET, CMD_MAP_10(mem_addr));
|
|
return 0;
|
|
|
|
case ONENAND_CMD_ERASE_VERIFY:
|
|
s3c_write_cmd(ONENAND_ERASE_VERIFY, CMD_MAP_10(mem_addr));
|
|
return 0;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static unsigned char *s3c_get_bufferram(struct mtd_info *mtd, int area)
|
|
{
|
|
struct onenand_chip *this = mtd->priv;
|
|
int index = ONENAND_CURRENT_BUFFERRAM(this);
|
|
unsigned char *p;
|
|
|
|
if (area == ONENAND_DATARAM) {
|
|
p = (unsigned char *) onenand->page_buf;
|
|
if (index == 1)
|
|
p += this->writesize;
|
|
} else {
|
|
p = (unsigned char *) onenand->oob_buf;
|
|
if (index == 1)
|
|
p += mtd->oobsize;
|
|
}
|
|
|
|
return p;
|
|
}
|
|
|
|
static int onenand_read_bufferram(struct mtd_info *mtd, loff_t addr, int area,
|
|
unsigned char *buffer, int offset,
|
|
size_t count)
|
|
{
|
|
unsigned char *p;
|
|
|
|
p = s3c_get_bufferram(mtd, area);
|
|
memcpy(buffer, p + offset, count);
|
|
return 0;
|
|
}
|
|
|
|
static int onenand_write_bufferram(struct mtd_info *mtd, loff_t addr, int area,
|
|
const unsigned char *buffer, int offset,
|
|
size_t count)
|
|
{
|
|
unsigned char *p;
|
|
|
|
p = s3c_get_bufferram(mtd, area);
|
|
memcpy(p + offset, buffer, count);
|
|
return 0;
|
|
}
|
|
|
|
static int s3c_onenand_bbt_wait(struct mtd_info *mtd, int state)
|
|
{
|
|
struct samsung_onenand *reg = (struct samsung_onenand *)onenand->base;
|
|
unsigned int flags = INT_ACT | LOAD_CMP;
|
|
unsigned int stat;
|
|
unsigned long timeout = 0x10000;
|
|
|
|
while (timeout--) {
|
|
stat = readl(®->int_err_stat);
|
|
if (stat & flags)
|
|
break;
|
|
}
|
|
/* To get correct interrupt status in timeout case */
|
|
stat = readl(&onenand->reg->int_err_stat);
|
|
writel(stat, &onenand->reg->int_err_ack);
|
|
|
|
if (stat & LD_FAIL_ECC_ERR) {
|
|
s3c_onenand_reset();
|
|
return ONENAND_BBT_READ_ERROR;
|
|
}
|
|
|
|
if (stat & LOAD_CMP) {
|
|
int ecc = readl(&onenand->reg->ecc_err_stat);
|
|
if (ecc & ONENAND_ECC_4BIT_UNCORRECTABLE) {
|
|
s3c_onenand_reset();
|
|
return ONENAND_BBT_READ_ERROR;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void s3c_onenand_check_lock_status(struct mtd_info *mtd)
|
|
{
|
|
struct onenand_chip *this = mtd->priv;
|
|
unsigned int block, end;
|
|
|
|
end = this->chipsize >> this->erase_shift;
|
|
|
|
for (block = 0; block < end; block++) {
|
|
s3c_read_cmd(CMD_MAP_01(onenand->mem_addr(block, 0, 0)));
|
|
|
|
if (readl(&onenand->reg->int_err_stat) & LOCKED_BLK) {
|
|
printf("block %d is write-protected!\n", block);
|
|
writel(LOCKED_BLK, &onenand->reg->int_err_ack);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void s3c_onenand_do_lock_cmd(struct mtd_info *mtd, loff_t ofs,
|
|
size_t len, int cmd)
|
|
{
|
|
struct onenand_chip *this = mtd->priv;
|
|
int start, end, start_mem_addr, end_mem_addr;
|
|
|
|
start = ofs >> this->erase_shift;
|
|
start_mem_addr = onenand->mem_addr(start, 0, 0);
|
|
end = start + (len >> this->erase_shift) - 1;
|
|
end_mem_addr = onenand->mem_addr(end, 0, 0);
|
|
|
|
if (cmd == ONENAND_CMD_LOCK) {
|
|
s3c_write_cmd(ONENAND_LOCK_START, CMD_MAP_10(start_mem_addr));
|
|
s3c_write_cmd(ONENAND_LOCK_END, CMD_MAP_10(end_mem_addr));
|
|
} else {
|
|
s3c_write_cmd(ONENAND_UNLOCK_START, CMD_MAP_10(start_mem_addr));
|
|
s3c_write_cmd(ONENAND_UNLOCK_END, CMD_MAP_10(end_mem_addr));
|
|
}
|
|
|
|
this->wait(mtd, FL_LOCKING);
|
|
}
|
|
|
|
static void s3c_onenand_unlock_all(struct mtd_info *mtd)
|
|
{
|
|
struct onenand_chip *this = mtd->priv;
|
|
loff_t ofs = 0;
|
|
size_t len = this->chipsize;
|
|
|
|
/* FIXME workaround */
|
|
this->subpagesize = mtd->writesize;
|
|
mtd->subpage_sft = 0;
|
|
|
|
if (this->options & ONENAND_HAS_UNLOCK_ALL) {
|
|
/* Write unlock command */
|
|
this->command(mtd, ONENAND_CMD_UNLOCK_ALL, 0, 0);
|
|
|
|
/* No need to check return value */
|
|
this->wait(mtd, FL_LOCKING);
|
|
|
|
/* Workaround for all block unlock in DDP */
|
|
if (!ONENAND_IS_DDP(this)) {
|
|
s3c_onenand_check_lock_status(mtd);
|
|
return;
|
|
}
|
|
|
|
/* All blocks on another chip */
|
|
ofs = this->chipsize >> 1;
|
|
len = this->chipsize >> 1;
|
|
}
|
|
|
|
s3c_onenand_do_lock_cmd(mtd, ofs, len, ONENAND_CMD_UNLOCK);
|
|
s3c_onenand_check_lock_status(mtd);
|
|
}
|
|
|
|
int s5pc110_chip_probe(struct mtd_info *mtd)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int s5pc210_chip_probe(struct mtd_info *mtd)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
void s3c_onenand_init(struct mtd_info *mtd)
|
|
{
|
|
struct onenand_chip *this = mtd->priv;
|
|
u32 size = (4 << 10); /* 4 KiB */
|
|
|
|
onenand = malloc(sizeof(struct s3c_onenand));
|
|
if (!onenand)
|
|
return;
|
|
|
|
onenand->page_buf = malloc(size * sizeof(char));
|
|
if (!onenand->page_buf)
|
|
return;
|
|
memset(onenand->page_buf, 0xff, size);
|
|
|
|
onenand->oob_buf = malloc(128 * sizeof(char));
|
|
if (!onenand->oob_buf)
|
|
return;
|
|
memset(onenand->oob_buf, 0xff, 128);
|
|
|
|
onenand->mtd = mtd;
|
|
|
|
#if defined(CONFIG_S5P)
|
|
onenand->base = (void *)0xE7100000;
|
|
onenand->ahb_addr = (void *)0xB0000000;
|
|
#endif
|
|
onenand->mem_addr = s3c_mem_addr;
|
|
onenand->reg = (struct samsung_onenand *)onenand->base;
|
|
|
|
this->read_word = s3c_onenand_readw;
|
|
this->write_word = s3c_onenand_writew;
|
|
|
|
this->wait = s3c_onenand_wait;
|
|
this->bbt_wait = s3c_onenand_bbt_wait;
|
|
this->unlock_all = s3c_onenand_unlock_all;
|
|
this->command = s3c_onenand_command;
|
|
|
|
this->read_bufferram = onenand_read_bufferram;
|
|
this->write_bufferram = onenand_write_bufferram;
|
|
|
|
this->options |= ONENAND_RUNTIME_BADBLOCK_CHECK;
|
|
}
|