blob: a70d8b792beeb81b8a39e2d616f2a7b37597faaa [file] [log] [blame]
/* SPDX-License-Identifier: (GPL-2.0+ OR MIT) */
/*
* drivers/nand/dev/cmd_amlnf.c
*
* Copyright (C) 2020 Amlogic, Inc. All rights reserved.
*
*/
#include <config.h>
#include <common.h>
#include <command.h>
#include "../include/amlnf_dev.h"
#ifdef AML_NAND_UBOOT
extern void amlnf_disprotect(char * name);
extern struct amlnand_phydev *aml_phy_get_dev(char * name);
extern struct amlnf_dev* aml_nftl_get_dev(char * name);
extern void amlnf_get_chip_size(u64 *size);
extern void show_phydev_list(void);
extern void show_ldev_list(void);
#endif
extern void amlnf_dump_chipinfo(void);
extern int roomboot_nand_read(struct amlnand_phydev *phydev);
extern int roomboot_nand_write(struct amlnand_phydev *phydev);
extern int nand_read_ops(struct amlnand_phydev *phydev);
extern int nand_write_ops(struct amlnand_phydev *phydev);
extern int nand_erase(struct amlnand_phydev *phydev);
extern void amlnand_dump_page(struct amlnand_phydev *phydev);
extern int amlnf_erase_ops(uint64_t off, uint64_t erase_len,unsigned char scrub_flag);
extern int amlnf_markbad_reserved_ops(uint32_t start_blk);
#if (AML_CFG_DTB_RSV_EN)
extern int amlnf_dtb_save(u8 *buf, int len);
extern int amlnf_dtb_read(u8 *buf, int len);
extern int amlnf_dtb_erase(void);
#endif
#if (AML_CFG_KEY_RSV_EN)
extern int amlnf_key_write(u8 *buf, int len, uint32_t *actual_lenth);
extern int amlnf_key_read(u8 * buf, int len, uint32_t *actual_lenth);
extern int amlnf_key_erase(void);
#endif
extern int amlnf_init(unsigned char flag);
extern int amlnf_exit(void);
//static int plane_mode;
struct amlnf_dev * nftl_device = NULL;
struct amlnand_phydev *phy_device=NULL;
static int nand_protect = 1;
static inline int isstring(char *p)
{
char *endptr = p;
while (*endptr != '\0') {
if (!(((*endptr >= '0') && (*endptr <= '9'))
|| ((*endptr >= 'a') && (*endptr <= 'f'))
|| ((*endptr >= 'A') && (*endptr <= 'F'))
|| (*endptr == 'x') || (*endptr == 'X')))
return 1;
endptr++;
}
return 0;
}
#ifdef AML_NAND_UBOOT
/*
* repack.
**/
unsigned long strtoul(const char *cp, char **endp,
unsigned int base)
{
return simple_strtoul(cp, endp, base);
}
#endif /* AML_NAND_UBOOT */
static inline int str2long(char *p, ulong *num)
{
char *endptr;
*num = strtoul(p, &endptr, 16);
return (*p != '\0' && *endptr == '\0') ? 1 : 0;
}
static inline int str2longlong(char *p, u64 *num)
{
char *endptr;
#ifndef AML_NAND_UBOOT
*num = strtoull(p, &endptr, 16);
#else
*num = simple_strtoul(p, &endptr, 16);
#endif
if (*endptr != '\0') {
switch (*endptr) {
case 'g':
case 'G':
*num <<= 10;
case 'm':
case 'M':
*num <<= 10;
case 'k':
case 'K':
*num <<= 10;
endptr++;
break;
}
}
return (*p != '\0' && *endptr == '\0') ? 1 : 0;
}
static int arg_off_size(int argc, char *argv[],
u64 chipsize,
u64 *off,
u64 *size)
{
if (argc >= 1) {
if (!(str2longlong(argv[0], (u64 *)off))) {
aml_nand_dbg("'%s' is not a number", argv[0]);
return -1;
}
} else
*off = 0;
if (argc >= 2) {
if (!(str2longlong(argv[1], (u64 *)size))) {
aml_nand_dbg("'%s' is not a number", argv[1]);
return -1;
}
} else
*size = chipsize - *off;
if (*size == chipsize)
aml_nand_dbg("whole chip/dev");
else
aml_nand_dbg("offset 0x%llx, size 0x%llx", *off, *size);
return 0;
}
#define AML_NFTL_ALIGN_SIZE 512
#define AML_NFTL_ALIGN_SHIFT 9
u8 local_buf[AML_NFTL_ALIGN_SIZE];
int amlnand_read(struct amlnf_dev *nftl_dev,
u8 *buf,
u64 offset, /* in bytes */
u64 size) /* in bytes */
{
u32 ret = 0;
u64 head_sector;
u64 head_start_bytes;
u64 head_bytes_num;
u64 mid_sector;
u64 mid_len;
u64 tail_sector;
u64 tail_bytes_num;
mid_len = offset >> AML_NFTL_ALIGN_SHIFT; /* in sector */
head_start_bytes = offset - (mid_len << AML_NFTL_ALIGN_SHIFT);
head_bytes_num = AML_NFTL_ALIGN_SIZE - head_start_bytes;
head_sector = offset >> AML_NFTL_ALIGN_SHIFT;
CMD_LINE
if (head_bytes_num >= size) {
CMD_LINE
ret |= nftl_dev->read_sector(nftl_dev,
head_sector,
1,
local_buf);
memcpy(buf, local_buf+head_start_bytes, size);
return ret;
}
CMD_LINE
if ((offset % AML_NFTL_ALIGN_SIZE) != 0) {
ret |= nftl_dev->read_sector(nftl_dev, head_sector, 1, local_buf);
memcpy(buf, local_buf+head_start_bytes, head_bytes_num);
CMD_LINE
buf += head_bytes_num;
offset += head_bytes_num;
size -= head_bytes_num;
}
if (size > AML_NFTL_ALIGN_SIZE) {
mid_len = size >> AML_NFTL_ALIGN_SHIFT;
mid_sector = offset >> AML_NFTL_ALIGN_SHIFT;
CMD_LINE
ret |= nftl_dev->read_sector(nftl_dev,
mid_sector,
mid_len,
buf);
buf += mid_len << AML_NFTL_ALIGN_SHIFT;
offset += mid_len << AML_NFTL_ALIGN_SHIFT;
size = size - (mid_len << AML_NFTL_ALIGN_SHIFT);
}
if (size == 0)
return ret;
CMD_LINE
tail_sector = offset >> AML_NFTL_ALIGN_SHIFT;
tail_bytes_num = size;
ret |= nftl_dev->read_sector(nftl_dev, tail_sector, 1, local_buf);
memcpy(buf, local_buf, tail_bytes_num);
CMD_LINE
return ret;
}
int amlnand_write(struct amlnf_dev *nftl_dev,
u8 *buf,
u64 offset,
u64 size)
{
u32 ret = 0;
u64 head_sector;
u64 head_start_bytes;
u64 head_bytes_num;
u64 mid_sector;
u64 mid_len;
u64 tail_sector;
u64 tail_bytes_num;
CMD_LINE
mid_len = offset >> AML_NFTL_ALIGN_SHIFT;
head_start_bytes = offset - (mid_len << AML_NFTL_ALIGN_SHIFT);
head_bytes_num = AML_NFTL_ALIGN_SIZE - head_start_bytes;
head_sector = offset >> AML_NFTL_ALIGN_SHIFT;
CMD_LINE
if (head_bytes_num >= size) {
ret |= nftl_dev->read_sector(nftl_dev,
head_sector,
1,
local_buf);
memcpy(local_buf+head_start_bytes, buf, size);
ret |= nftl_dev->write_sector(nftl_dev,
head_sector,
1,
local_buf);
goto flush;
}
CMD_LINE
if ((offset % AML_NFTL_ALIGN_SIZE) != 0) { //sectore alignment
ret |= nftl_dev->read_sector(nftl_dev, head_sector, 1, local_buf);
memcpy(local_buf+head_start_bytes, buf, head_bytes_num);
ret |= nftl_dev->write_sector(nftl_dev, head_sector, 1, local_buf);
buf += head_bytes_num;
offset += head_bytes_num;
size -= head_bytes_num;
}
if (size > AML_NFTL_ALIGN_SIZE) { //why 4
CMD_LINE
mid_len = size >> AML_NFTL_ALIGN_SHIFT;
mid_sector = offset >> AML_NFTL_ALIGN_SHIFT;
ret |= nftl_dev->write_sector(nftl_dev,
mid_sector,
mid_len,
buf);
buf += mid_len << AML_NFTL_ALIGN_SHIFT;
offset += mid_len << AML_NFTL_ALIGN_SHIFT;
size = size - (mid_len << AML_NFTL_ALIGN_SHIFT);
}
if (size == 0)
goto flush;
CMD_LINE
tail_sector = offset >> AML_NFTL_ALIGN_SHIFT;
tail_bytes_num = size;
ret |= nftl_dev->read_sector(nftl_dev, tail_sector, 1, local_buf);
memcpy(local_buf, buf, tail_bytes_num);
ret |= nftl_dev->write_sector(nftl_dev, tail_sector, 1, local_buf);
flush:
CMD_LINE
ret = nftl_dev->flush((struct amlnf_dev *)nftl_dev);
if (ret < 0) {
aml_nand_msg("nftl flush cache failed");
ret = -1;
}
CMD_LINE
return ret;
}
extern void dbg_phyop(void);
static int do_amlnfphy(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
{
struct amlnand_phydev *phydev = NULL;
struct phydev_ops *devops = NULL;
struct amlnf_dev *nftl_dev;
unsigned long addr;
u64 chipsize, erase_addr, erase_len, erase_off, off, size;
u8 devread, nfread_flag;
int start_secor, length, ret = 0;
char *cmd, *protect_name;
char *dev_name = NULL;
unsigned long markbad_reserved_addr = 0;
/* at least two arguments please */
if (argc < 1)
goto usage;
cmd = argv[1];
/* for dbg entry! */
if (strcmp(cmd, "dbg") == 0) {
dbg_phyop();
return 0;
}
/* show boot flag!*/
if (strcmp(cmd, "boot") == 0) {
aml_nand_msg("device_boot_flag 0x%x", device_boot_flag);
if (device_boot_flag == NAND_BOOT_FLAG) {
aml_nand_msg("boot from nand!");
} else {
aml_nand_msg("bootflag not ready yet!");
}
return 0;
}
if (strcmp(cmd, "device") == 0) {
//printk("argc %d\n", argc);
if (argc == 2) {
show_phydev_list();
return 0;
}
phydev = aml_phy_get_dev(dev_name);
if (!phydev) {
aml_nand_msg("phydev be NULL, can't get it!");
return -1;
}
return 0;
}
if (strcmp(cmd, "env") == 0) {
aml_nand_dbg("env relocate");
env_relocate();
return 0;
}
if (strcmp(cmd, "disprotect") == 0) {
protect_name = argv[2];
/* fixme, using amlnf_chip*/
amlnf_disprotect(protect_name);
return 0;
}
if (strcmp(cmd, "exit") == 0) {
amlnf_exit();
return 0;
}
if (strcmp(cmd, "init") == 0) {
putc('\n');
int init_flag = (ulong)strtoul(argv[2], NULL, 16);
/* flag = 0, indicate normal boot; */
/* flag = 1, indicate update, with data; */
/* flag = 2, indicate need erase */
aml_nand_msg("init_flag:%x",init_flag);
ret = amlnf_init(init_flag);
if (ret) {
aml_nand_msg("nand_init failed ret:%x", ret);
return ret;
}
phydev = aml_phy_get_dev(NAND_CODE_NAME);
if (!phydev) {
aml_nand_msg("phydev be NULL");
goto usage;
}
return 0;
}
//ldevice
if (strcmp(cmd, "ldevice") == 0) {
//printk("argc %d\n", argc);
if (argc == 2) {
show_ldev_list();
return 0;
}
//fixme,
goto usage;
return 0;
}
if ((strcmp(cmd, "read_byte") == 0)
|| (strcmp(cmd, "write_byte") == 0)
|| (strcmp(cmd, "lwrite") == 0)
||(strcmp(cmd, "lread") == 0)) {
if (argc < 6)
goto usage;
dev_name = argv[2];
nftl_dev = NULL;
nftl_dev = aml_nftl_get_dev(dev_name);
if (!nftl_dev) {
aml_nand_msg("nftl_dev be NULL");
return -1;
}
aml_nand_dbg("nftl_dev->name =%s", nftl_dev->name);
addr = (ulong)strtoul(argv[3], NULL, 16);
nfread_flag = 0;
if ((strncmp(cmd, "read_byte", 9) == 0)
||(strncmp(cmd, "lread", 5) == 0) )
nfread_flag = 1;
aml_nand_dbg("\nNAND %s: addr:%llx ",
nfread_flag ? "read_byte" : "write_byte",
addr);
if (arg_off_size(argc - 4, (char **)(argv + 4), 0x0, &off, &size) != 0) {
goto usage;
}
if (nfread_flag)
ret = amlnand_read(nftl_dev, (u8 *)addr, off, size);
else
ret = amlnand_write(nftl_dev, (u8 *)addr, off, size);
aml_nand_dbg(" 0x%llx bytes %s : %s",
size,
nfread_flag ? "read_byte" : "write_byte",
ret ? "ERROR" : "OK");
return ret;
}
if ((strcmp(cmd, "read") == 0) || (strcmp(cmd, "write") == 0)) {
if (argc < 6)
goto usage;
dev_name = argv[2];
nftl_dev = NULL;
nftl_dev = aml_nftl_get_dev(dev_name);
if (!nftl_dev) {
aml_nand_msg("nftl_dev be NULL");
return -1;
}
aml_nand_dbg("nftl_dev->nand_dev->writesize =%x",
nftl_dev->nand_dev->writesize);
aml_nand_dbg("nftl_dev->nand_dev->erasesize =%x",
nftl_dev->nand_dev->erasesize);
aml_nand_dbg("nftl_dev->name =%s", nftl_dev->name);
addr = (ulong)strtoul(argv[3], NULL, 16);
nfread_flag = strncmp(cmd, "read", 4) == 0;
/* 1 = read, 0 = write */
aml_nand_msg("NAND %s: addr:%lx",
nfread_flag ? "read" : "write",
addr);
if (arg_off_size(argc - 4, (char **)(argv + 4), 0x0, &off, &size) != 0)
goto usage;
if (off % 512) {
start_secor = ((int) (off / 512) + 1);
aml_nand_dbg("secor+1");
} else
start_secor = (int) (off / 512);
if (size % 512) {
length = ((int)((size / 512))+1);
aml_nand_dbg("length+1");
} else
length = (int)((size / 512));
aml_nand_dbg("start_secor =%d", start_secor);
aml_nand_dbg("length_sector =%d", length);
if (nfread_flag) {
ret = nftl_dev->read_sector(nftl_dev,
start_secor,
length,
(u8 *)addr);
if (ret < 0) {
aml_nand_dbg("read %d sector failed", length);
return -1;
}
} else {
ret = nftl_dev->write_sector(nftl_dev,
start_secor,
length,
(u8 *)addr);
if (ret < 0) {
aml_nand_dbg("write %d sector failed", length);
return -1;
}
ret = nftl_dev->flush(nftl_dev);
if (ret < 0) {
aml_nand_dbg("nftl flush cache failed");
return -1;
}
}
aml_nand_msg(" %d sector %s : %s",
length,
nfread_flag ? "read" : "write",
ret ? "ERROR" : "OK");
return ret;
}
if (strcmp(cmd, "chipinfo") == 0) {
putc('\n');
amlnf_dump_chipinfo();
return 0;
}
if (strcmp(cmd, "size") == 0) {
if (argc < 4)
goto usage;
dev_name = argv[2];
nftl_dev = NULL;
nftl_dev = aml_nftl_get_dev(dev_name);
if (!nftl_dev) {
aml_nand_msg("nftl_dev be NULL");
return -1;
}
aml_nand_dbg("nftl_dev->nand_dev->writesize =%x",
nftl_dev->nand_dev->writesize);
aml_nand_dbg("nftl_dev->nand_dev->erasesize =%x",
nftl_dev->nand_dev->erasesize);
aml_nand_dbg("nftl_dev->name =%s", nftl_dev->name);
addr = (ulong)strtoul(argv[3], NULL, 16);
*(u64 *)addr = nftl_dev->size_sector;
aml_nand_dbg("nftl_dev->size_sector =%llx",
nftl_dev->size_sector);
aml_nand_dbg("*addr = %llx", (*(u64 *)addr));
return 0;
}
if (strncmp(cmd, "rom_protect", 9) == 0) {
if (argc < 2)
goto usage;
if (strncmp(argv[2], "on", 2) == 0)
nand_protect = 1;
else if (strncmp(argv[2], "off", 3) == 0)
nand_protect = 0;
else
goto usage;
return 0;
}
if ((strcmp(cmd, "rom_write") == 0)
|| (strcmp(cmd, "rom_read") == 0)) {
nfread_flag = 0;
if (strncmp(cmd, "rom_read", 8) == 0)
nfread_flag = 1;
if (argc < 4)
goto usage;
addr = (ulong)strtoul(argv[2], NULL, 16);
aml_nand_msg("AMLNAND %s: ",
nfread_flag ? "rom_read" : "rom_write");
struct amlnand_phydev *tmp_phydev = phydev;
struct phydev_ops *tmp_devops = devops;
phydev = aml_phy_get_dev(NAND_BOOT_NAME);
if (!phydev) {
aml_nand_msg("phydev be NULL");
return -1;
}
devops = &(phydev->ops);
aml_nand_dbg("phydev->name =%s", phydev->name);
amldev_dumpinfo(phydev);
if (arg_off_size(argc - 3,
(char **)(argv + 3),
phydev->size,
&off,
&size) != 0)
return -1;
memset(devops, 0x0, sizeof(struct phydev_ops));
devops->addr = off;
devops->len = size;
devops->mode = NAND_HW_ECC;
devops->datbuf = (u8 *)addr;
if (nfread_flag) {
ret = roomboot_nand_read(phydev);
if (ret < 0)
aml_nand_msg("nand read uboot failed");
} else {
ret = roomboot_nand_write(phydev);
if (ret < 0)
aml_nand_msg("nand write uboot failed");
}
aml_nand_msg("%llu bytes %s : %s",
size,
nfread_flag ? "rom_read" : "rom_write",
ret ? "ERROR" : "OK");
phydev = tmp_phydev;
devops = tmp_devops;
return ret;
}
#if (AML_CFG_DTB_RSV_EN)
if ((strcmp(cmd, "dtb_write") == 0)
|| (strcmp(cmd, "dtb_read") == 0)) {
nfread_flag = 0;
if (strncmp(cmd, "dtb_read", 8) == 0)
nfread_flag = 1;
if (argc < 3)
goto usage;
addr = (ulong)strtoul(argv[2], NULL, 16);
size = (ulong)strtoul(argv[3], NULL, 16);
aml_nand_msg("cmd %s: ",
nfread_flag ? "dtb_read" : "dtb_write");
//memset(devops, 0x0, sizeof(struct phydev_ops));
if (nfread_flag) {
ret = amlnf_dtb_read((u8 *)addr, (int)size);
if (ret < 0)
aml_nand_msg("nand read dtd failed");
} else {
ret = amlnf_dtb_save((u8 *)addr, (int)size);
if (ret < 0)
aml_nand_msg("nand write dtd failed");
}
aml_nand_msg("%llu bytes %s : %s",
size,
nfread_flag ? "dtd_read" : "dtd_write",
ret ? "ERROR" : "OK");
return ret;
}
if (strcmp(cmd, "dtb_erase") == 0) {
ret = amlnf_dtb_erase();
aml_nand_msg("dtb erase %s", ret ? "Fail" : "Okay");
return ret;
}
#endif
/* need full environments */
#if (AML_CFG_KEY_RSV_EN)
uint32_t actual_lenth;
if ((strcmp(cmd, "key_write") == 0)
|| (strcmp(cmd, "key_read") == 0)) {
nfread_flag = 0;
if (strncmp(cmd, "key_read", 8) == 0)
nfread_flag = 1;
if (argc < 3)
goto usage;
addr = (ulong)strtoul(argv[2], NULL, 16);
size = (ulong)strtoul(argv[3], NULL, 16);
aml_nand_msg("cmd %s: ",
nfread_flag ? "key_read" : "key_write");
//memset(devops, 0x0, sizeof(struct phydev_ops));
if (nfread_flag) {
ret = amlnf_key_read((u8 *)addr, (int)size, &actual_lenth);
if (ret < 0)
aml_nand_msg("nand read key failed");
} else {
ret = amlnf_key_write((u8 *)addr, (int)size, &actual_lenth);
if (ret < 0)
aml_nand_msg("nand write key failed");
}
aml_nand_msg("%llu bytes %s : %s",
size,
nfread_flag ? "key_read" : "key_write",
ret ? "ERROR" : "OK");
return ret;
}
if (strcmp(cmd, "key_erase") == 0) {
ret = amlnf_key_erase();
aml_nand_msg("key erase %s", ret ? "Fail" : "Okay");
return ret;
}
#endif
/* avoid fail... */
amlnf_get_chip_size(&chipsize);
if ((strcmp(cmd, "devread") == 0) || (strcmp(cmd, "devwrite") == 0)) {
if (argc < 6)
goto usage;
dev_name = argv[2];
if (strcmp(dev_name, "boot") == 0)
dev_name = NAND_BOOT_NAME;
else if (strcmp(dev_name, "code") == 0)
dev_name = NAND_CODE_NAME;
else if (strcmp(dev_name, "cache") == 0)
dev_name = NAND_CACHE_NAME;
else if (strcmp(dev_name, "data") == 0)
dev_name = NAND_DATA_NAME;
else {
aml_nand_msg("input wrong name!! %s", dev_name);
goto usage;
}
addr = (ulong)strtoul(argv[3], NULL, 16);
aml_nand_dbg("addr = %llx", addr);
devread = strncmp(cmd, "devread", 7) == 0;
/* 1 = devread, 0 = devwrite */
aml_nand_msg("NAND %s: addr:%lx",
devread ? "devread" : "devwrite",
addr);
if (arg_off_size(argc - 4,
(char **)(argv + 4),
chipsize,
&off,
&size) != 0)
goto usage;
phydev = aml_phy_get_dev(dev_name);
if (!phydev) {
aml_nand_msg("phydev be NULL");
return -1;
}
devops = &(phydev->ops);
memset(devops, 0x0, sizeof(struct phydev_ops));
devops->addr = off;
devops->len = size;
devops->mode = NAND_HW_ECC;
devops->datbuf = (u8 *)addr;
if (devread) {
ret = nand_read_ops(phydev);
if (ret < 0)
aml_nand_dbg("nand read failed");
} else {
ret = nand_write_ops(phydev);
if (ret < 0)
aml_nand_dbg("nand write failed");
}
aml_nand_msg("%llu bytes %s : %s",
size,
devread ? "devread" : "devwrite",
ret ? "ERROR" : "OK");
return 0;
}
if ((strcmp(cmd, "deverase") == 0)) {
if (argc < 4)
goto usage;
int percent = 0;
int percent_complete = -1;
int without_check = 0;
dev_name = argv[2];
if (strcmp(dev_name, "boot") == 0) {
dev_name = NAND_BOOT_NAME;
/* do not check bad block in uboot area! */
without_check = 1;
}
else if (strcmp(dev_name, "code") == 0)
dev_name = NAND_CODE_NAME;
else if (strcmp(dev_name, "cache") == 0)
dev_name = NAND_CACHE_NAME;
else if (strcmp(dev_name, "data") == 0)
dev_name = NAND_DATA_NAME;
else {
aml_nand_msg("input wrong name!! %s", dev_name);
goto usage;
}
phydev = aml_phy_get_dev(dev_name);
if (!phydev) {
aml_nand_msg("phydev be NULL");
return -1;
}
devops = &(phydev->ops);
if (!strcmp(argv[3], "whole")) {
off = 0;
size = phydev->size;
erase_addr = erase_off = off;
erase_len = size;
printf("whole dev.\n");
} else {
if ((strcmp(cmd, "deverase") == 0) && (argc < 3))
goto usage;
if ((arg_off_size(argc - 3,
(char **)(argv + 3),
phydev->size,
&off,
&size) != 0))
goto usage;
aml_nand_dbg("off:0x%llx size:%llx.\n", off, size);
erase_addr = erase_off = off;
erase_len = size;
}
aml_nand_dbg("erase_len = %llx", erase_len);
aml_nand_dbg("erase_off = %llx", erase_off);
if (erase_len < phydev->erasesize) {
aml_nand_msg("size 0x%016llx smaller than one blk 0x%08x",
erase_len,
phydev->erasesize);
aml_nand_msg("Erasing 0x%08x instead",
phydev->erasesize);
erase_len = phydev->erasesize;
}
for (;
erase_addr < erase_off + erase_len;
erase_addr += phydev->erasesize) {
memset(devops, 0x0, sizeof(struct phydev_ops));
devops->addr = erase_addr;
devops->len = phydev->erasesize;
devops->mode = NAND_HW_ECC;
if (!without_check) {
ret = phydev->block_isbad(phydev);
if (ret > 0) {
aml_nand_msg("\rSkipping bad block at 0x%08llx",
erase_addr);
continue;
} else if (ret < 0) {
aml_nand_msg("get blk failed:ret=%d addr=%llx",
ret,
erase_addr);
return -1;
}
}
ret = nand_erase(phydev);
if (ret < 0) {
aml_nand_msg("\nAMLNAND Erase fail:%d %llx\n",
ret,
erase_addr);
ret = phydev->block_markbad(phydev);
if (ret < 0)
aml_nand_msg("bad blk mark fail:%llx\n",
erase_addr);
continue;
}
percent = (erase_addr * 100) / (erase_off + erase_len);
if ((percent != percent_complete)
&& ((percent % 10) == 0)) {
percent_complete = percent;
aml_nand_msg("erasing %d %%-%d %% complete",
percent,
percent+10);
}
}
aml_nand_msg("NAND %s %s\n", "ERASE",
(ret < 0) ? "ERROR" : "OK");
return 0;
}
if (strcmp(cmd, "dump") == 0) {
dev_name = argv[2];
if (strcmp(dev_name, "boot") == 0)
dev_name = NAND_BOOT_NAME;
else if (strcmp(dev_name, "cache") == 0)
dev_name = NAND_CACHE_NAME;
else if (strcmp(dev_name, "code") == 0)
dev_name = NAND_CODE_NAME;
else if (strcmp(dev_name, "data") == 0)
dev_name = NAND_DATA_NAME;
else {
aml_nand_msg("input wrong name!! %s", dev_name);
goto usage;
}
addr = (ulong)strtoul(argv[3], NULL, 16);
aml_nand_dbg("addr = %llx", addr);
if (arg_off_size(argc - 4,
(char **)(argv + 4),
chipsize,
&off,
&size) != 0)
goto usage;
phydev = aml_phy_get_dev(dev_name);
if (!phydev) {
aml_nand_msg("phydev be NULL");
return -1;
}
devops = &(phydev->ops);
memset(devops, 0x0, sizeof(struct phydev_ops));
devops->addr = off;
devops->len = phydev->writesize;
devops->oobbuf = NULL;
devops->datbuf = (u8 *)addr;
devops->mode = NAND_SOFT_ECC;
amlnand_dump_page(phydev);
return 0;
}
if ((strcmp(cmd, "scrub") == 0) || (strcmp(cmd, "erase") == 0)) {
int scrub_flag = !strncmp(cmd, "scrub", 5);
if (argc < 2)
goto usage;
if (scrub_flag) {
puts("Warning:"
"devscrub option will erase all factory set "\
"bad blocks !\n"\
" "\
"There is no reliable way to recover them.\n"\
" "\
"Use this command only for testing purposes "\
"if you\n"\
" "\
"are sure of what you are doing !\n"
"\nReally scrub this NAND flash ? < y/N >"\
"\n");
scrub_flag = 0;
if (nand_protect) {
if (getc() == 'y') {
puts("y");
if (getc() == '\r')
scrub_flag = 1;
else {
puts("scrub aborted\n");
return -1;
}
} else {
puts("scrub aborted\n");
return -1;
}
} else
scrub_flag = 1;
}
if (!strcmp(argv[2], "whole")) {
off = 0;
/* ((u64)flash->chipsize << 20); */
size = chipsize;
erase_addr = erase_off = off;
erase_len = size;
printf("whole dev.\n");
} else {
if ((arg_off_size(argc - 2,
(char **)(argv + 2),
chipsize,
&off,
&size) != 0))
goto usage;
erase_addr = erase_off = off;
erase_len = size;
}
erase_addr = erase_off = off;
erase_len = size;
ret = amlnf_erase_ops(off, erase_len, scrub_flag);
if (ret < 0)
aml_nand_msg("nand erase failed");
return ret;
}
if (strcmp(cmd, "markbad") == 0) {
if (argc < 4) {
goto usage;
}
dev_name = argv[2];
if (strcmp(dev_name, "boot") == 0) {
dev_name = NAND_BOOT_NAME;
}
else if(strcmp(dev_name, "code") == 0){
dev_name = NAND_CODE_NAME;
}else if(strcmp(dev_name, "cache") == 0){
dev_name = NAND_CACHE_NAME;
}else if(strcmp(dev_name, "data") == 0){
dev_name = NAND_DATA_NAME;
}else{
aml_nand_msg("input wrong name!! %s",dev_name);
goto usage;
}
phydev = aml_phy_get_dev(dev_name);
if (!phydev) {
aml_nand_msg("phydev be NULL");
return -1;
}
devops = &(phydev->ops);
if ((arg_off_size(argc - 3, (char **)(argv + 3), phydev->size, &off, &size) != 0)) {
goto usage;
}
aml_nand_dbg("off:0x%llx size:%llx.\n", off, size);
erase_addr =erase_off= off;
erase_len = size;
aml_nand_dbg("erase_len = %llx",erase_len);
aml_nand_dbg("erase_off = %llx",erase_off);
if (erase_len < phydev->erasesize) {
printf("Warning: markbad size 0x%08x smaller than one " \
"block 0x%08x\n",(unsigned int)erase_len, phydev->erasesize);
printf(" markbad 0x%08x instead\n", phydev->erasesize);
erase_len = phydev->erasesize;
}
for (; erase_addr <erase_off + erase_len; erase_addr += phydev->erasesize) {
memset(devops, 0x0, sizeof(struct phydev_ops));
devops->addr = erase_addr;
devops->len = phydev->erasesize;
devops->mode = NAND_HW_ECC;
ret = phydev->block_isbad(phydev);
if (ret > 0) {
printf("\rSkipping bad block at 0x%08llx\n", erase_addr);
continue;
} else if (ret < 0) {
printf("\n:AMLNAND get bad block failed: ret=%d at addr=%llx\n",ret, erase_addr);
return -1;
}
ret = phydev->block_markbad(phydev);
if (ret < 0)
printf("AMLNAND bad block mark failed: %llx\n", erase_addr);
}
printf("NAND %s %s\n", "MARKBAD", (ret <0) ? "ERROR" : "OK");
return 0;
}
if (strcmp(cmd, "markbad_reserved") == 0) {
if (argc < 3) {
goto usage;
}
if (!(str2long(argv[2], (unsigned long*)&markbad_reserved_addr))) {
aml_nand_dbg("'%s' is not a number", argv[2]);
goto usage;
}
printf("mark_reserved block:%d\n", (int)markbad_reserved_addr);
ret = amlnf_markbad_reserved_ops(markbad_reserved_addr);
printf("NAND %s %s\n", "MARKBAD", (ret <0) ? "ERROR" : "OK");
return 0;
}
usage:
printf("may be invalid argus, check again.\n");
cmd_usage(cmdtp);
return 1;
}
#ifdef CONFIG_SYS_LONGHELP
static char amlnand_help_text[] =
"init - init amlnand_phy here\n"
"chipinfo - show aml chip information\n"
"device[dev] - show or set current device\n"
"partition* [part] - show or set current partition\n"
"plane[dev] - show or set current plane mode\n"
"read - addr off|partition size\n"
"write - addr off|partition size\n"
" read/write 'size' bytes starting at offset 'off'\n"
" to/from memory address 'addr', skipping bad blocks.\n"
"erase[clean|whole][off size] - erase 'size' bytes from\n"
" offset 'off' (entire device if not specified)\n"
"dump addr off\n"
" show the raw data to addr at offset off\n"
"read_byte - addr off|partition size\n"
"write_byte - addr off|partition size\n"
" read_byte/write_byte 'size' bytes starting at offset 'off'\n"
" to/from memory address 'addr', skipping bad blocks.\n"
"devread - addr off|partition size\n"
"devwrite - addr off|partition size\n"
" read/write 'size' bytes starting at offset 'off' in device[dev]\n"
" to/from memory address 'addr', skipping bad blocks.\n"
"deverase[whole][off size] - erase 'size' bytes from\n"
" offset 'off' (entire device if not specified) in device[dev]\n"
"markbad addr - mark block bad at addr\n"
"mark_reserved reserved_blk_NO -mark reserved_blk_NO bad \n"
"ldevice[dev] - show/get nftl(logic) device by name\n"
"rom_read/write addr off cnt - read/write uboot.\n"
"boot - show boot flag"
#if (AML_CFG_DTB_RSV_EN)
"dtb_read/write addr cnt - read/write dtd.\n"
"dtb_erase - erase dtb!\n"
#endif
#if (AML_CFG_KEY_RSV_EN)
"key_read/write addr cnt - read/write dtd.\n"
"key_erase - erase keys!\n"
#endif
"";
#endif
U_BOOT_CMD(
amlnf, CONFIG_SYS_MAXARGS, 0, do_amlnfphy,
"aml nand sub-system",
amlnand_help_text
);