1473 lines
46 KiB
C
1473 lines
46 KiB
C
/*
|
|
* Copyright (c) 2023, Intel Corporation.
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*/
|
|
#include "flash_cadence_nand_ll.h"
|
|
|
|
LOG_MODULE_REGISTER(flash_cdns_nand_ll, CONFIG_FLASH_LOG_LEVEL);
|
|
|
|
/**
|
|
* Wait for the Cadence NAND controller to become idle.
|
|
*
|
|
* @param base_address The base address of the Cadence NAND controller.
|
|
* @retval 0 on success or -ETIMEDOUT error value on failure.
|
|
*/
|
|
static inline int32_t cdns_nand_wait_idle(uintptr_t base_address)
|
|
{
|
|
/* Wait status command response ready */
|
|
if (!WAIT_FOR(CNF_GET_CTRL_BUSY(sys_read32(CNF_CMDREG(base_address, CTRL_STATUS))) == 0U,
|
|
IDLE_TIME_OUT, k_msleep(1))) {
|
|
LOG_ERR("Timed out waiting for wait idle response");
|
|
return -ETIMEDOUT;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Set the row address for a NAND flash memory device using the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param local_row_address The row address.
|
|
* @param page_set The page set number.
|
|
*/
|
|
static void row_address_set(struct cadence_nand_params *params, uint32_t *local_row_address,
|
|
uint32_t page_set)
|
|
{
|
|
uint32_t block_number = 0;
|
|
|
|
block_number = ((page_set) / (params->npages_per_block));
|
|
*local_row_address = 0;
|
|
*local_row_address |= ROW_VAL_SET((params->page_size_bit) - 1, 0,
|
|
((page_set) % (params->npages_per_block)));
|
|
*local_row_address |=
|
|
ROW_VAL_SET((params->block_size_bit) - 1, (params->page_size_bit), block_number);
|
|
*local_row_address |= ROW_VAL_SET((params->lun_size_bit) - 1, (params->block_size_bit),
|
|
(block_number / params->nblocks_per_lun));
|
|
}
|
|
|
|
/**
|
|
* Retrieve information about the NAND flash device using the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @retval 0 on success or -ENXIO error value on failure.
|
|
*/
|
|
static int cdns_nand_device_info(struct cadence_nand_params *params)
|
|
{
|
|
struct nf_ctrl_version *nf_ver;
|
|
uintptr_t base_address;
|
|
uint32_t reg_value = 0;
|
|
uint8_t type;
|
|
|
|
base_address = params->nand_base;
|
|
|
|
/* Read flash device version information */
|
|
reg_value = sys_read32(CNF_CTRLPARAM(base_address, VERSION));
|
|
nf_ver = (struct nf_ctrl_version *)®_value;
|
|
|
|
LOG_INF("NAND Flash Version Information");
|
|
LOG_INF("HPNFC Magic Number 0x%x", nf_ver->hpnfc_magic_number);
|
|
LOG_INF("Fixed number 0x%x", nf_ver->ctrl_fix);
|
|
LOG_INF("Controller Revision Number 0x%x", nf_ver->ctrl_rev);
|
|
|
|
/* Interface Type */
|
|
reg_value = sys_read32(CNF_CTRLPARAM(base_address, DEV_PARAMS0));
|
|
type = CNF_GET_DEV_TYPE(reg_value);
|
|
if (type == CNF_DT_UNKNOWN) {
|
|
LOG_ERR("%s: device type unknown", __func__);
|
|
return -ENXIO;
|
|
}
|
|
|
|
params->nluns = CNF_GET_NLUNS(reg_value);
|
|
LOG_INF("Number of LUMs %hhx", params->nluns);
|
|
|
|
/* Pages per block */
|
|
reg_value = sys_read32(CNF_CTRLCFG(base_address, DEV_LAYOUT));
|
|
params->npages_per_block = GET_PAGES_PER_BLOCK(reg_value);
|
|
|
|
/* Page size and spare size */
|
|
reg_value = sys_read32(CNF_CTRLPARAM(base_address, DEV_AREA));
|
|
params->page_size = GET_PAGE_SIZE(reg_value);
|
|
params->spare_size = GET_SPARE_SIZE(reg_value);
|
|
|
|
/* Device blocks per LUN */
|
|
params->nblocks_per_lun = sys_read32(CNF_CTRLPARAM(base_address, DEV_BLOCKS_PLUN));
|
|
|
|
/* Calculate block size and total device size */
|
|
params->block_size = (params->npages_per_block * params->page_size);
|
|
params->device_size = ((long long)params->block_size *
|
|
(long long)(params->nblocks_per_lun * params->nluns));
|
|
LOG_INF("block size %x total device size %llx", params->block_size, params->device_size);
|
|
|
|
/* Calculate bit size of page, block and lun*/
|
|
params->page_size_bit = find_msb_set((params->npages_per_block) - 1);
|
|
params->block_size_bit = find_msb_set((params->nblocks_per_lun) - 1);
|
|
params->lun_size_bit = find_msb_set((params->nluns) - 1);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Retrieve the status of a specific thread in the Cadence NAND controller.
|
|
*
|
|
* @param base_address The base address of the Cadence NAND controller.
|
|
* @param thread The thread identifier.
|
|
* @retval The status of the thread.
|
|
*/
|
|
static uint32_t cdns_nand_get_thrd_status(uintptr_t base_address, uint8_t thread)
|
|
{
|
|
uint32_t status;
|
|
|
|
sys_write32(THREAD_VAL(thread), (base_address + CMD_STATUS_PTR_ADDR));
|
|
status = sys_read32((base_address + CMD_STAT_CMD_STATUS));
|
|
return status;
|
|
}
|
|
|
|
/**
|
|
* Wait for a specific thread in the Cadence controller to complete.
|
|
*
|
|
* @param base_address The base address of the Cadence controller.
|
|
* @param thread The thread identifier to wait for.
|
|
* @retval 0 on success or -ETIMEDOUT error value on failure.
|
|
*/
|
|
static int cdns_wait_for_thread(uintptr_t base_address, uint8_t thread)
|
|
{
|
|
|
|
if (!WAIT_FOR((sys_read32((base_address) + THR_STATUS) & BIT(thread)) == 0U,
|
|
THREAD_IDLE_TIME_OUT, k_msleep(1))) {
|
|
LOG_ERR("Timed out waiting for thread response");
|
|
return -ETIMEDOUT;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Set features in the Cadence NAND controller using PIO operations.
|
|
*
|
|
* @param base_address The base address of the Cadence NAND controller.
|
|
* @param feat_addr The address of the feature to be set.
|
|
* @param feat_val The value of the feature to be set.
|
|
* @param thread The thread identifier for the PIO operation.
|
|
* @param vol_id The volume identifier for the feature set operation.
|
|
* @param use_intr Flag indicating whether to use interrupts during the operation.
|
|
* @retval 0 on success or -ETIMEDOUT error value on failure.
|
|
*/
|
|
static int cdns_nand_pio_set_features(uintptr_t base_address, uint8_t feat_addr, uint8_t feat_val,
|
|
uint8_t thread, uint8_t vol_id)
|
|
{
|
|
uint32_t status = 0;
|
|
int ret = 0;
|
|
|
|
ret = cdns_wait_for_thread(base_address, thread);
|
|
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
sys_write32(SET_FEAT_ADDR(feat_addr), (base_address + CDNS_CMD_REG1));
|
|
sys_write32(feat_val, (base_address + CDNS_CMD_REG2));
|
|
status = CMD_0_THREAD_POS_SET(thread);
|
|
status |= CMD_0_C_MODE_SET(CT_PIO_MODE);
|
|
status |= PIO_CMD0_CT_SET(PIO_SET_FEA_MODE);
|
|
status |= CMD_0_VOL_ID_SET(vol_id);
|
|
sys_write32(status, (base_address + CDNS_CMD_REG0));
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Check whether a transfer complete for PIO operation in the Cadence controller has finished.
|
|
*
|
|
* @param base_address The base address of the Cadence controller.
|
|
* @param thread The thread identifier for the PIO operation.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
static int cdns_pio_transfer_complete(uintptr_t base_address, uint8_t thread)
|
|
{
|
|
uint32_t status;
|
|
|
|
status = WAIT_FOR(((cdns_nand_get_thrd_status(base_address, thread)) != 0), IDLE_TIME_OUT,
|
|
k_msleep(1));
|
|
|
|
if (status == 0) {
|
|
LOG_ERR("Timed out waiting for thread status response");
|
|
return -ETIMEDOUT;
|
|
}
|
|
|
|
if ((status & (BIT(F_CSTAT_COMP)))) {
|
|
if ((status & (BIT(F_CSTAT_FAIL)))) {
|
|
LOG_ERR("Cadence status operation failed %s", __func__);
|
|
return -EIO;
|
|
}
|
|
} else {
|
|
LOG_ERR("Cadence status complete failed %s", __func__);
|
|
return -EIO;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Set the operational mode for the Cadence NAND controller.
|
|
*
|
|
* @param base_address The base address of the Cadence NAND controller.
|
|
* @param opr_mode The operational mode SDR / NVDDR to set.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
static int cdns_nand_set_opr_mode(uintptr_t base_address, uint8_t opr_mode)
|
|
{
|
|
uint8_t device_type;
|
|
uint32_t timing_mode = 0;
|
|
uint32_t status;
|
|
int ret;
|
|
|
|
if (opr_mode == CNF_OPR_WORK_MODE_SDR) {
|
|
|
|
status = ONFI_TIMING_MODE_SDR(
|
|
sys_read32(CNF_CTRLPARAM(base_address, ONFI_TIMING_0)));
|
|
timing_mode = find_lsb_set(status) - 1;
|
|
|
|
/* PHY Register Timing setting*/
|
|
sys_write32(PHY_CTRL_REG_SDR, (base_address + PHY_CTRL_REG_OFFSET));
|
|
sys_write32(PHY_TSEL_REG_SDR, (base_address + PHY_TSEL_REG_OFFSET));
|
|
sys_write32(PHY_DQ_TIMING_REG_SDR, (base_address + PHY_DQ_TIMING_REG_OFFSET));
|
|
sys_write32(PHY_DQS_TIMING_REG_SDR, (base_address + PHY_DQS_TIMING_REG_OFFSET));
|
|
sys_write32(PHY_GATE_LPBK_CTRL_REG_SDR, (base_address + PHY_GATE_LPBK_OFFSET));
|
|
sys_write32(PHY_DLL_MASTER_CTRL_REG_SDR, (base_address + PHY_DLL_MASTER_OFFSET));
|
|
|
|
/* Async mode timing settings */
|
|
sys_write32((CNF_ASYNC_TIMINGS_TRH) | (CNF_ASYNC_TIMINGS_TRP) |
|
|
(CNF_ASYNC_TIMINGS_TWH) | (CNF_ASYNC_TIMINGS_TWP),
|
|
CNF_MINICTRL(base_address, ASYNC_TOGGLE_TIMINGS));
|
|
|
|
/* Set operation work mode in common settings */
|
|
sys_clear_bits(CNF_MINICTRL(base_address, CMN_SETTINGS),
|
|
CNF_OPR_WORK_MODE_SDR_MASK);
|
|
|
|
} else {
|
|
/* NVDDR MODE */
|
|
status = ONFI_TIMING_MODE_NVDDR(
|
|
sys_read32(CNF_CTRLPARAM(base_address, ONFI_TIMING_0)));
|
|
timing_mode = find_lsb_set(status) - 1;
|
|
/* PHY Register Timing setting*/
|
|
sys_write32(PHY_CTRL_REG_DDR, (base_address + PHY_CTRL_REG_OFFSET));
|
|
sys_write32(PHY_TSEL_REG_DDR, (base_address + PHY_TSEL_REG_OFFSET));
|
|
sys_write32(PHY_DQ_TIMING_REG_DDR, (base_address + PHY_DQ_TIMING_REG_OFFSET));
|
|
sys_write32(PHY_DQS_TIMING_REG_DDR, (base_address + PHY_DQS_TIMING_REG_OFFSET));
|
|
sys_write32(PHY_GATE_LPBK_CTRL_REG_DDR, (base_address + PHY_GATE_LPBK_OFFSET));
|
|
sys_write32(PHY_DLL_MASTER_CTRL_REG_DDR, (base_address + PHY_DLL_MASTER_OFFSET));
|
|
/* Set operation work mode in common settings */
|
|
sys_set_bits(CNF_MINICTRL(base_address, CMN_SETTINGS),
|
|
CNF_OPR_WORK_MODE_NVDDR_MASK);
|
|
}
|
|
|
|
/* Wait for controller to be in idle state */
|
|
ret = cdns_nand_wait_idle(base_address);
|
|
if (ret != 0) {
|
|
LOG_ERR("Wait for controller to be in idle state Failed");
|
|
return ret;
|
|
}
|
|
|
|
/* Check device type */
|
|
device_type = CNF_GET_DEV_TYPE(sys_read32(CNF_CTRLPARAM(base_address, DEV_PARAMS0)));
|
|
|
|
if (device_type != ONFI_INTERFACE) {
|
|
LOG_ERR("Driver does not support this interface");
|
|
return -ENOTSUP;
|
|
}
|
|
/* Reset DLL PHY */
|
|
sys_clear_bit(CNF_MINICTRL(base_address, DLL_PHY_CTRL), CNF_DLL_PHY_RST_N);
|
|
|
|
/* Wait for controller to be in idle state */
|
|
ret = cdns_nand_wait_idle(base_address);
|
|
if (ret != 0) {
|
|
LOG_ERR("Wait for controller to be in idle state Failed");
|
|
return ret;
|
|
}
|
|
|
|
ret = cdns_nand_pio_set_features(base_address, SET_FEAT_TIMING_MODE_ADDRESS, timing_mode,
|
|
NF_TDEF_TRD_NUM, VOL_ID);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
ret = cdns_pio_transfer_complete(base_address, NF_TDEF_TRD_NUM);
|
|
if (ret != 0) {
|
|
LOG_ERR("cdns pio check failed");
|
|
return ret;
|
|
}
|
|
|
|
ret = cdns_nand_wait_idle(base_address);
|
|
if (ret != 0) {
|
|
LOG_ERR("Wait for controller to be in idle state Failed");
|
|
return ret;
|
|
}
|
|
|
|
/* set dll_rst_n in dll_phy_ctrl to 1 */
|
|
sys_set_bit(CNF_MINICTRL(base_address, DLL_PHY_CTRL), CNF_DLL_PHY_RST_N);
|
|
|
|
ret = cdns_nand_wait_idle(base_address);
|
|
if (ret != 0) {
|
|
LOG_ERR("Wait for controller to be in idle state Failed");
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Configure the transfer settings of the Cadence NAND controller.
|
|
*
|
|
* @param base_address The base address of the Cadence NAND controller.
|
|
* @retval 0 on success or -ETIMEDOUT error value on failure.
|
|
*/
|
|
static int cdns_nand_transfer_config(uintptr_t base_address)
|
|
{
|
|
int ret = 0;
|
|
/* Wait for controller to be in idle state */
|
|
ret = cdns_nand_wait_idle(base_address);
|
|
|
|
if (ret != 0) {
|
|
LOG_ERR("Wait for controller to be in idle state Failed");
|
|
return ret;
|
|
}
|
|
|
|
/* Configure data transfer parameters */
|
|
sys_write32(ENABLE, CNF_CTRLCFG(base_address, TRANS_CFG0));
|
|
|
|
/* Disable cache and multiplane. */
|
|
sys_write32(DISABLE, CNF_CTRLCFG(base_address, MULTIPLANE_CFG));
|
|
sys_write32(DISABLE, CNF_CTRLCFG(base_address, CACHE_CFG));
|
|
|
|
/* Clear all interrupts. */
|
|
sys_write32(CLEAR_ALL_INTERRUPT, (base_address + INTR_STATUS));
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Initialize the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
int cdns_nand_init(struct cadence_nand_params *params)
|
|
{
|
|
uint32_t reg_value_read = 0;
|
|
uintptr_t base_address = params->nand_base;
|
|
uint8_t datarate_mode = params->datarate_mode;
|
|
int ret;
|
|
|
|
if (!WAIT_FOR(CNF_GET_INIT_COMP(sys_read32(CNF_CMDREG(base_address, CTRL_STATUS))) != 0U,
|
|
IDLE_TIME_OUT, k_msleep(1))) {
|
|
LOG_ERR("Timed out waiting for NAND Controller Init complete status response");
|
|
return -ETIMEDOUT;
|
|
}
|
|
|
|
if (CNF_GET_INIT_FAIL(sys_read32(CNF_CMDREG(base_address, CTRL_STATUS))) != 0) {
|
|
LOG_ERR("NAND Controller Init complete Failed!!!");
|
|
return -ENODEV;
|
|
}
|
|
|
|
ret = cdns_nand_device_info(params);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
/* Hardware Support Features */
|
|
reg_value_read = sys_read32(CNF_CTRLPARAM(base_address, FEATURE));
|
|
/* Enable data integrity parity check if the data integrity parity mechanism is */
|
|
/* supported by the device */
|
|
if (CNF_HW_DI_PR_SUPPORT(reg_value_read) != 0) {
|
|
sys_set_bit(CNF_DI(base_address, CONTROL), CNF_DI_PAR_EN);
|
|
}
|
|
|
|
/* Enable data integrity CRC check if the data integrity CRC mechanism is */
|
|
/* supported by the device */
|
|
if (CNF_HW_DI_CRC_SUPPORT(reg_value_read) != 0) {
|
|
sys_set_bit(CNF_DI(base_address, CONTROL), CNF_DI_CRC_EN);
|
|
}
|
|
/* Status polling mode, device control and status register */
|
|
ret = cdns_nand_wait_idle(base_address);
|
|
|
|
if (ret != 0) {
|
|
LOG_ERR("Wait for controller to be in idle state Failed");
|
|
return ret;
|
|
}
|
|
sys_write32(DEV_STAT_DEF_VALUE, CNF_CTRLCFG(base_address, DEV_STAT));
|
|
|
|
/* Set operation work mode */
|
|
ret = cdns_nand_set_opr_mode(base_address, datarate_mode);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
/* Set data transfer configuration parameters */
|
|
ret = cdns_nand_transfer_config(base_address);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
/* Wait for controller to be in idle state */
|
|
ret = cdns_nand_wait_idle(base_address);
|
|
if (ret != 0) {
|
|
LOG_ERR("Wait for controller to be in idle state Failed");
|
|
return ret;
|
|
}
|
|
|
|
/* DMA Setting */
|
|
sys_write32((F_BURST_SEL_SET(NF_TDEF_BURST_SEL)) | (BIT(F_OTE)),
|
|
(base_address + NF_DMA_SETTING));
|
|
|
|
/* Pre fetch */
|
|
sys_write32(((NF_FIFO_TRIGG_LVL_SET(PRE_FETCH_VALUE)) |
|
|
(NF_DMA_PACKAGE_SIZE_SET(PRE_FETCH_VALUE))),
|
|
(base_address + NF_PRE_FETCH));
|
|
/* Total bits in row addressing*/
|
|
params->total_bit_row = find_msb_set(((params->npages_per_block) - 1)) +
|
|
find_msb_set((params->nblocks_per_lun) - 1);
|
|
|
|
if (ret != 0) {
|
|
LOG_ERR("Failed to establish device access width!");
|
|
return -EINVAL;
|
|
}
|
|
/* Enable Global Interrupt for NAND*/
|
|
#ifdef CONFIG_CDNS_NAND_INTERRUPT_SUPPORT
|
|
sys_set_bit((base_address + INTERRUPT_STATUS_REG), GINTR_ENABLE);
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
#if CONFIG_CDNS_NAND_CDMA_MODE
|
|
/**
|
|
*
|
|
* This function performs Command descriptor structure prepareation.
|
|
*
|
|
* @param nf_mem determine which NF memory bank will be selected
|
|
* @param flash_ptr start ROW address in NF memory
|
|
* @param mem_ptr system memory pointer
|
|
* @param ctype Command type (read/write/erase)
|
|
* @param cmd_cnt counter for commands
|
|
* @param dma_sel select DMA engine (0 - slave DMA, 1 - master DMA)
|
|
* @param vol_id specify target volume ID
|
|
*
|
|
*/
|
|
void cdns_nand_cdma_prepare(char nf_mem, uint32_t flash_ptr, char *mem_ptr, uint16_t ctype,
|
|
int32_t cmd_cnt, uint8_t dma_sel, uint8_t vol_id,
|
|
struct cdns_cdma_command_descriptor *desc)
|
|
{
|
|
struct cdns_cdma_command_descriptor *cdma_desc;
|
|
|
|
cdma_desc = desc;
|
|
/* set fields for one descriptor */
|
|
cdma_desc->flash_pointer = flash_ptr;
|
|
cdma_desc->bank_number = nf_mem;
|
|
cdma_desc->command_flags |= CDMA_CF_DMA_MASTER_SET(dma_sel) | F_CFLAGS_VOL_ID_SET(vol_id);
|
|
cdma_desc->memory_pointer = (uintptr_t)mem_ptr;
|
|
cdma_desc->status = 0;
|
|
cdma_desc->sync_flag_pointer = 0;
|
|
cdma_desc->sync_arguments = 0;
|
|
cdma_desc->ctrl_data_ptr = 0x40;
|
|
cdma_desc->command_type = ctype;
|
|
if (cmd_cnt > 1) {
|
|
cdma_desc->next_pointer = (uintptr_t)(desc + 1);
|
|
cdma_desc->command_flags |= CFLAGS_MPTRPC_SET | CFLAGS_MPTRPC_SET;
|
|
cdma_desc->command_flags |= CFLAGS_CONT_SET;
|
|
} else {
|
|
cdma_desc->next_pointer = 0;
|
|
#ifdef CONFIG_CDNS_NAND_INTERRUPT_SUPPORT
|
|
cdma_desc->command_flags |= CDMA_CF_INT_SET;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Check a command descriptor transfer complete status in the Cadence NAND controller.
|
|
*
|
|
* @param desc_ptr The pointer to the command descriptor structure.
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
static int cdns_transfer_complete(struct cdns_cdma_command_descriptor *desc_ptr,
|
|
struct cadence_nand_params *params)
|
|
{
|
|
#ifdef CONFIG_CDNS_NAND_INTERRUPT_SUPPORT
|
|
uint32_t status = 0;
|
|
|
|
NAND_INT_SEM_TAKE(params);
|
|
sys_write32(NF_TDEF_TRD_NUM, (params->nand_base + CMD_STATUS_PTR_ADDR));
|
|
status = sys_read32((params->nand_base + CMD_STAT_CMD_STATUS));
|
|
if ((status & (BIT(F_CSTAT_COMP)))) {
|
|
if ((status & (BIT(F_CSTAT_FAIL)))) {
|
|
LOG_ERR("Cadence status operation failed %s", __func__);
|
|
return -EIO;
|
|
}
|
|
} else {
|
|
LOG_ERR("Cadence status complete failed %s", __func__);
|
|
return -EIO;
|
|
}
|
|
#else
|
|
ARG_UNUSED(params);
|
|
|
|
if (!WAIT_FOR(((desc_ptr->status & (BIT(F_CSTAT_COMP))) != 0), IDLE_TIME_OUT,
|
|
k_msleep(1))) {
|
|
LOG_ERR("Timed out waiting for thread status response");
|
|
return -ETIMEDOUT;
|
|
}
|
|
if ((desc_ptr->status & (BIT(F_CSTAT_FAIL))) != 0) {
|
|
LOG_ERR("Cadence status operation failed %s", __func__);
|
|
return -EIO;
|
|
}
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Send a command descriptor to the Cadence NAND controller for execution.
|
|
*
|
|
* @param base_address The base address of the Cadence NAND controller.
|
|
* @param desc_ptr The pointer to the command descriptor.
|
|
* @param thread The thread number for the execution.
|
|
* @retval 0 on success or -ETIMEDOUT error value on failure.
|
|
*/
|
|
static int cdns_nand_send(uintptr_t base_address, char *desc_ptr, uint8_t thread)
|
|
{
|
|
uint64_t desc_address;
|
|
uint32_t status;
|
|
int ret;
|
|
|
|
desc_address = (uint64_t)desc_ptr;
|
|
|
|
ret = cdns_wait_for_thread(base_address, thread);
|
|
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
/* desc_ptr address passing */
|
|
sys_write32(desc_address & U32_MASK_VAL, (base_address + CDNS_CMD_REG2));
|
|
sys_write32((desc_address >> 32) & U32_MASK_VAL, (base_address + CDNS_CMD_REG3));
|
|
/* Thread selection */
|
|
status = CMD_0_THREAD_POS_SET(thread);
|
|
/* CDMA Mode selection */
|
|
status |= CMD_0_C_MODE_SET(CT_CDMA_MODE);
|
|
/* CMD 0 Reg write*/
|
|
sys_write32(status, (base_address + CDNS_CMD_REG0));
|
|
return 0;
|
|
}
|
|
|
|
static int cdns_cdma_desc_transfer_finish(struct cadence_nand_params *params, uint32_t page_count,
|
|
uint32_t max_page_desc, uint32_t ctype,
|
|
uint32_t cond_start, char *buffer)
|
|
{
|
|
uint32_t page_count_pass = 0;
|
|
uint32_t row_address = 0;
|
|
uint32_t base_address;
|
|
uint32_t page_buffer_size;
|
|
struct cdns_cdma_command_descriptor *cdma_desc;
|
|
int ret;
|
|
|
|
page_buffer_size = (page_count > max_page_desc) ? max_page_desc : page_count;
|
|
|
|
cdma_desc = k_malloc(sizeof(struct cdns_cdma_command_descriptor) * page_buffer_size);
|
|
|
|
if (cdma_desc == NULL) {
|
|
LOG_ERR("Memory allocation error occurred %s", __func__);
|
|
return -ENOSR;
|
|
}
|
|
|
|
base_address = params->nand_base;
|
|
|
|
while (page_count > 0) {
|
|
row_address_set(params, &row_address, cond_start);
|
|
|
|
if (page_count > max_page_desc) {
|
|
page_count_pass = max_page_desc;
|
|
page_count = page_count - max_page_desc;
|
|
cond_start = cond_start + page_count_pass;
|
|
} else {
|
|
page_count_pass = page_count;
|
|
page_count = page_count - page_count_pass;
|
|
}
|
|
for (int index = 0; index < page_count_pass; index++) {
|
|
cdns_nand_cdma_prepare(NF_TDEF_DEV_NUM, row_address, buffer,
|
|
(ctype + index), (page_count_pass - index),
|
|
DMA_MS_SEL, VOL_ID, (cdma_desc + index));
|
|
}
|
|
ret = cdns_nand_send(base_address, (char *)cdma_desc, NF_TDEF_TRD_NUM);
|
|
|
|
if (ret != 0) {
|
|
k_free(cdma_desc);
|
|
return ret;
|
|
}
|
|
|
|
if (ctype != CNF_CMD_ERASE) {
|
|
buffer = buffer + (max_page_desc * params->page_size);
|
|
}
|
|
|
|
ret = cdns_transfer_complete(cdma_desc, params);
|
|
|
|
if (ret != 0) {
|
|
k_free(cdma_desc);
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
k_free(cdma_desc);
|
|
|
|
return 0;
|
|
}
|
|
/**
|
|
* Perform a CDMA write operation for the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param start_page_number The starting page number for the write operation.
|
|
* @param buffer The buffer containing the data to be written.
|
|
* @param page_count The number of pages to be written.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
static int cdns_nand_cdma_write(struct cadence_nand_params *params, uint32_t start_page_number,
|
|
char *buffer, uint32_t page_count)
|
|
{
|
|
int ret;
|
|
|
|
ret = cdns_cdma_desc_transfer_finish(params, page_count, CONFIG_FLASH_CDNS_CDMA_PAGE_COUNT,
|
|
CNF_CMD_WR, start_page_number, buffer);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* Perform a CDMA read operation for the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param start_page_number The starting page number for the read operation.
|
|
* @param buffer The buffer to store the read data.
|
|
* @param page_count The number of pages to be read.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
static int cdns_nand_cdma_read(struct cadence_nand_params *params, uint32_t start_page_number,
|
|
char *buffer, uint32_t page_count)
|
|
{
|
|
int ret;
|
|
|
|
ret = cdns_cdma_desc_transfer_finish(params, page_count, CONFIG_FLASH_CDNS_CDMA_PAGE_COUNT,
|
|
CNF_CMD_RD, start_page_number, buffer);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* Perform a CDMA erase operation for the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param start_block_number The starting block number for the erase operation.
|
|
* @param block_count The number of blocks to be erased.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
static int cdns_nand_cdma_erase(struct cadence_nand_params *params, uint32_t start_block_number,
|
|
uint32_t block_count)
|
|
{
|
|
int ret;
|
|
|
|
ret = cdns_cdma_desc_transfer_finish(params, block_count,
|
|
CONFIG_FLASH_CDNS_CDMA_BLOCK_COUNT, CNF_CMD_ERASE,
|
|
start_block_number, NULL);
|
|
|
|
return ret;
|
|
}
|
|
#endif
|
|
|
|
#if CONFIG_CDNS_NAND_PIO_MODE
|
|
|
|
/**
|
|
* Perform an erase operation on the Cadence NAND controller using PIO.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param thread The thread identifier for the PIO operation.
|
|
* @param bank The bank identifier for the erase operation.
|
|
* @param start_block The starting block number for the erase operation.
|
|
* @param ctype The command type for the erase operation.
|
|
* @param block_count The number of blocks to be erased.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
static int cdns_nand_pio_erase(struct cadence_nand_params *params, uint8_t thread, uint8_t bank,
|
|
uint32_t start_block, uint16_t ctype, uint32_t block_count)
|
|
{
|
|
uint32_t status;
|
|
uintptr_t base_address;
|
|
uint32_t row_address = 0;
|
|
uint32_t index = 0;
|
|
int ret;
|
|
|
|
base_address = params->nand_base;
|
|
for (index = 0; index < block_count; index++) {
|
|
|
|
ret = cdns_wait_for_thread(base_address, thread);
|
|
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
row_address_set(params, &row_address, (start_block * params->npages_per_block));
|
|
sys_write32(row_address, (base_address + CDNS_CMD_REG1));
|
|
start_block++;
|
|
sys_write32((NF_CMD4_BANK_SET(bank)), (base_address + CDNS_CMD_REG4));
|
|
status = CMD_0_THREAD_POS_SET(thread);
|
|
#ifdef CONFIG_CDNS_NAND_INTERRUPT_SUPPORT
|
|
status |= BIT(PIO_CF_INT);
|
|
#endif
|
|
status |= CMD_0_C_MODE_SET(CT_PIO_MODE);
|
|
status |= PIO_CMD0_CT_SET(ctype);
|
|
sys_write32(status, (base_address + CDNS_CMD_REG0));
|
|
NAND_INT_SEM_TAKE(params);
|
|
ret = cdns_pio_transfer_complete(base_address, thread);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Prepare for a PIO operation in the Cadence NAND controller.
|
|
*
|
|
* @param base_address The base address of the Cadence NAND controller.
|
|
* @param thread The thread ID associated with the operation.
|
|
* @param bank The bank ID for the operation.
|
|
* @param row_address The row address for the operation.
|
|
* @param buf The buffer containing the data for the operation.
|
|
* @param ctype The command type for the operation.
|
|
* @param dma_sel The DMA selection flag for the operation.
|
|
* @param vol_id The volume ID for the operation.
|
|
* @retval 0 on success or -ETIMEDOUT error value on failure.
|
|
*/
|
|
static int cdns_nand_pio_prepare(uintptr_t base_address, uint8_t thread, uint8_t bank,
|
|
uint32_t row_address, char *buf, uint16_t ctype, uint8_t dma_sel,
|
|
uint8_t vol_id)
|
|
{
|
|
uint64_t buf_addr = (uintptr_t)buf;
|
|
uint32_t status;
|
|
int ret;
|
|
|
|
ret = cdns_wait_for_thread(base_address, thread);
|
|
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
sys_write32(row_address, (base_address + CDNS_CMD_REG1));
|
|
sys_write32(NF_CMD4_BANK_SET(bank), (base_address + CDNS_CMD_REG4));
|
|
sys_write32(buf_addr & U32_MASK_VAL, (base_address + CDNS_CMD_REG2));
|
|
sys_write32((buf_addr >> 32) & U32_MASK_VAL, (base_address + CDNS_CMD_REG3));
|
|
status = CMD_0_THREAD_POS_SET(thread);
|
|
#ifdef CONFIG_CDNS_NAND_INTERRUPT_SUPPORT
|
|
status |= PIO_CF_INT_SET;
|
|
#endif
|
|
status |= PIO_CF_DMA_MASTER_SET(dma_sel);
|
|
status |= CMD_0_C_MODE_SET(CT_PIO_MODE);
|
|
status |= PIO_CMD0_CT_SET(ctype);
|
|
status |= CMD_0_VOL_ID_SET(vol_id);
|
|
sys_write32(status, (base_address + CDNS_CMD_REG0));
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Perform a PIO write operation for the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param row_address The row address for the write operation.
|
|
* @param buffer The buffer containing the data to be written.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
static int cdns_nand_pio_write(struct cadence_nand_params *params, uint32_t row_address,
|
|
char *buffer)
|
|
{
|
|
uintptr_t base_address;
|
|
int ret;
|
|
|
|
base_address = params->nand_base;
|
|
|
|
ret = cdns_nand_pio_prepare(base_address, NF_TDEF_TRD_NUM, NF_TDEF_DEV_NUM, row_address,
|
|
buffer, CNF_CMD_WR, DMA_MS_SEL, VOL_ID);
|
|
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
NAND_INT_SEM_TAKE(params);
|
|
ret = cdns_pio_transfer_complete(base_address, NF_TDEF_TRD_NUM);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Perform a PIO read operation for the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param row_address The row address for the read operation.
|
|
* @param buffer The buffer to store the read data.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
static int cdns_nand_pio_read(struct cadence_nand_params *params, uint32_t row_address,
|
|
char *buffer)
|
|
{
|
|
uintptr_t base_address;
|
|
int ret;
|
|
|
|
base_address = params->nand_base;
|
|
|
|
ret = cdns_nand_pio_prepare(base_address, NF_TDEF_TRD_NUM, NF_TDEF_DEV_NUM, row_address,
|
|
buffer, CNF_CMD_RD, DMA_MS_SEL, VOL_ID);
|
|
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
NAND_INT_SEM_TAKE(params);
|
|
ret = cdns_pio_transfer_complete(base_address, NF_TDEF_TRD_NUM);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Perform a combined PIO read and write operation for the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param start_page_number The starting page number for the read/write operation.
|
|
* @param buffer The buffer containing the data to be written or to store the read data.
|
|
* @param page_count The number of pages to be read or written.
|
|
* @param mode The mode of operation (read, write).
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
static int cdns_nand_pio_read_write(struct cadence_nand_params *params, uint32_t start_page_number,
|
|
char *buffer, uint32_t page_count, uint8_t mode)
|
|
{
|
|
uint32_t index;
|
|
uint32_t pio_row_address = 0;
|
|
int ret = 0;
|
|
|
|
for (index = 0; index < page_count; index++) {
|
|
row_address_set(params, &pio_row_address, start_page_number++);
|
|
if (mode == CDNS_READ) {
|
|
ret = cdns_nand_pio_read(params, pio_row_address,
|
|
buffer + (index * (params->page_size)));
|
|
} else {
|
|
ret = cdns_nand_pio_write(params, pio_row_address,
|
|
buffer + (index * (params->page_size)));
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
#endif
|
|
|
|
#if CONFIG_CDNS_NAND_GENERIC_MODE
|
|
/**
|
|
* Send a generic command to the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param mini_ctrl_cmd The command to be sent.
|
|
* @retval 0 on success or -ETIMEDOUT error value on failure.
|
|
*/
|
|
static int cdns_generic_send_cmd(struct cadence_nand_params *params, uint64_t mini_ctrl_cmd)
|
|
{
|
|
|
|
uint32_t mini_ctrl_cmd_l, mini_ctrl_cmd_h, status;
|
|
uintptr_t base_address;
|
|
int ret = 0;
|
|
|
|
base_address = params->nand_base;
|
|
mini_ctrl_cmd_l = mini_ctrl_cmd & U32_MASK_VAL;
|
|
mini_ctrl_cmd_h = mini_ctrl_cmd >> 32;
|
|
ret = cdns_nand_wait_idle(base_address);
|
|
|
|
if (ret != 0) {
|
|
LOG_ERR("Wait for controller to be in idle state Failed");
|
|
return ret;
|
|
}
|
|
sys_write32(mini_ctrl_cmd_l, (base_address + CDNS_CMD_REG2));
|
|
sys_write32(mini_ctrl_cmd_h, (base_address + CDNS_CMD_REG3));
|
|
/* Select generic command. */
|
|
status = CMD_0_THREAD_POS_SET(NF_TDEF_TRD_NUM);
|
|
#ifdef CONFIG_CDNS_NAND_INTERRUPT_SUPPORT
|
|
status |= GEN_CF_INT_SET(GEN_CF_INT_ENABLE);
|
|
#endif
|
|
status |= CMD_0_C_MODE_SET(CT_GENERIC_MODE);
|
|
sys_write32(status, (base_address + CDNS_CMD_REG0));
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Send a generic command data to the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param mode The mode of operation (read, write).
|
|
* @param data_length The length of the associated data.
|
|
* @retval 0 on success or -ETIMEDOUT error value on failure.
|
|
*/
|
|
static int cdns_generic_cmd_data(struct cadence_nand_params *params, uint8_t mode,
|
|
uint32_t data_length)
|
|
{
|
|
uint64_t mini_ctrl_cmd = 0;
|
|
int ret = 0;
|
|
|
|
mini_ctrl_cmd |= GCMD_TWB_VALUE;
|
|
mini_ctrl_cmd |= GCMCD_DATA_SEQ;
|
|
mini_ctrl_cmd |= GEN_SECTOR_COUNT_SET;
|
|
mini_ctrl_cmd |= GEN_LAST_SECTOR_SIZE_SET((uint64_t)data_length);
|
|
mini_ctrl_cmd |= GEN_DIR_SET((uint64_t)mode);
|
|
mini_ctrl_cmd |= GEN_SECTOR_SET((uint64_t)data_length);
|
|
ret = cdns_generic_send_cmd(params, mini_ctrl_cmd);
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* Wait for the completion of an SDMA operation in the Cadence NAND controller.
|
|
*
|
|
* @param base_address The base address of the Cadence NAND controller.
|
|
* @retval 0 on success or -ETIMEDOUT error value on failure.
|
|
*/
|
|
static int cdns_wait_sdma(uintptr_t base_address)
|
|
{
|
|
|
|
if (!WAIT_FOR(((sys_read32(base_address + INTR_STATUS) & BIT(SDMA_TRIGG)) != 0),
|
|
IDLE_TIME_OUT, k_msleep(1))) {
|
|
LOG_ERR("Timed out waiting for sdma response");
|
|
return -ETIMEDOUT;
|
|
}
|
|
sys_set_bit((base_address + INTR_STATUS), SDMA_TRIGG);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Perform buffer copying to SDMA regs in the Cadence NAND controller.
|
|
*
|
|
* @param sdma_base_address The base address of the SDMA in the Cadence NAND controller.
|
|
* @param buffer The source or destination buffer for the copy operation.
|
|
* @param data_length The length of the data to be copied.
|
|
*/
|
|
static void sdma_buffer_copy_in(uint32_t sdma_base_address, uint8_t *buffer, uint32_t data_length)
|
|
{
|
|
uint32_t index;
|
|
|
|
for (index = 0; index < data_length; index++) {
|
|
sys_write8(*(buffer + index), sdma_base_address + index);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Perform buffer copying from SDMA regs in the Cadence NAND controller.
|
|
*
|
|
* @param sdma_base_address The base address of the SDMA in the Cadence NAND controller.
|
|
* @param buffer The source or destination buffer for the copy operation.
|
|
* @param data_length The length of the data to be copied.
|
|
*/
|
|
static void sdma_buffer_copy_out(uint32_t sdma_base_address, uint8_t *buffer, uint32_t data_length)
|
|
{
|
|
uint32_t index;
|
|
|
|
for (index = 0; index < data_length; index++) {
|
|
*(buffer + index) = sys_read8(sdma_base_address + index);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Perform a generic page read operation in the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param read_address The address from which to read the page.
|
|
* @param data_buffer The buffer to store the read data.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
static int cdns_generic_page_read(struct cadence_nand_params *params, uint64_t read_address,
|
|
void *data_buffer)
|
|
{
|
|
uint64_t mini_ctrl_cmd = 0;
|
|
uintptr_t base_address = params->nand_base;
|
|
int ret;
|
|
|
|
mini_ctrl_cmd = PAGE_READ_CMD;
|
|
mini_ctrl_cmd |= GCMD_TWB_VALUE;
|
|
if ((params->nluns > 1) || (params->total_bit_row > 16)) {
|
|
mini_ctrl_cmd |= PAGE_MAX_BYTES(PAGE_MAX_SIZE);
|
|
} else {
|
|
mini_ctrl_cmd |= PAGE_MAX_BYTES(PAGE_MAX_SIZE - 1);
|
|
}
|
|
mini_ctrl_cmd |= read_address << 32;
|
|
ret = cdns_generic_send_cmd(params, mini_ctrl_cmd);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
NAND_INT_SEM_TAKE(params);
|
|
ret = cdns_generic_cmd_data(params, CDNS_READ, params->page_size);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
NAND_INT_SEM_TAKE(params);
|
|
ret = cdns_wait_sdma(base_address);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
sdma_buffer_copy_out(params->sdma_base, data_buffer, params->page_size);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Perform a generic page write operation in the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param write_address The address to which the page will be written.
|
|
* @param data_buffer The buffer containing the data to be written.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
static int cdns_generic_page_write(struct cadence_nand_params *params, uint64_t write_address,
|
|
void *data_buffer)
|
|
{
|
|
uint64_t mini_ctrl_cmd = 0;
|
|
int ret;
|
|
|
|
uintptr_t base_address = params->nand_base;
|
|
|
|
mini_ctrl_cmd |= GCMD_TWB_VALUE;
|
|
mini_ctrl_cmd |= GEN_ADDR_WRITE_DATA((uint32_t)write_address);
|
|
if ((params->nluns > 1) || (params->total_bit_row > BIT16_CHECK)) {
|
|
mini_ctrl_cmd |= PAGE_MAX_BYTES(PAGE_MAX_SIZE);
|
|
} else {
|
|
mini_ctrl_cmd |= PAGE_MAX_BYTES(PAGE_MAX_SIZE - 1);
|
|
}
|
|
mini_ctrl_cmd |= PAGE_WRITE_CMD;
|
|
ret = cdns_generic_send_cmd(params, mini_ctrl_cmd);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
NAND_INT_SEM_TAKE(params);
|
|
ret = cdns_generic_cmd_data(params, CDNS_WRITE, params->page_size);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
sdma_buffer_copy_in(params->sdma_base, data_buffer, params->page_size);
|
|
NAND_INT_SEM_TAKE(params);
|
|
mini_ctrl_cmd = 0;
|
|
mini_ctrl_cmd |= PAGE_WRITE_10H_CMD;
|
|
mini_ctrl_cmd |= GCMD_TWB_VALUE;
|
|
mini_ctrl_cmd |= PAGE_CMOD_CMD;
|
|
ret = cdns_generic_send_cmd(params, mini_ctrl_cmd);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
NAND_INT_SEM_TAKE(params);
|
|
ret = cdns_wait_sdma(base_address);
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* Perform a generic read or write operation for a range of pages in the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param start_page_number The starting page number for the read or write operation.
|
|
* @param buffer The buffer containing the data to be written or to store the read data.
|
|
* @param page_count The number of pages to be read or written.
|
|
* @param mode The mode of operation (read, write).
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
static int cdns_nand_gen_read_write(struct cadence_nand_params *params, uint32_t start_page_number,
|
|
char *buffer, uint32_t page_count, uint8_t mode)
|
|
{
|
|
uint64_t address = 0;
|
|
uint32_t index = 0;
|
|
uint32_t gen_row_address = 0;
|
|
int ret = 0;
|
|
|
|
for (index = 0; index < page_count; index++) {
|
|
row_address_set(params, &gen_row_address, start_page_number++);
|
|
address = ((uint64_t)gen_row_address);
|
|
if (mode == CDNS_READ) {
|
|
ret = cdns_generic_page_read(params, address,
|
|
buffer + (index * (params->page_size)));
|
|
if (ret != 0) {
|
|
LOG_ERR("Cadence NAND Generic Page Read Error!!");
|
|
return ret;
|
|
}
|
|
} else {
|
|
ret = cdns_generic_page_write(params, address,
|
|
buffer + (index * (params->page_size)));
|
|
if (ret != 0) {
|
|
LOG_ERR("Cadence NAND Generic Page write Error!!");
|
|
return ret;
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Perform a generic erase operation for a range of blocks in the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param start_block The starting block number for the erase operation.
|
|
* @param block_count The number of blocks to be erased.
|
|
* @retval 0 on success or -ETIMEDOUT error value on failure.
|
|
*/
|
|
static int cdns_nand_gen_erase(struct cadence_nand_params *params, uint32_t start_block,
|
|
uint32_t block_count)
|
|
{
|
|
uint64_t mini_ctrl_cmd = 0;
|
|
uintptr_t base_address = 0;
|
|
uint32_t gen_row_address = 0;
|
|
uint32_t index = 0;
|
|
int ret = 0;
|
|
|
|
for (index = 0; index < block_count; index++) {
|
|
row_address_set(params, &gen_row_address, (start_block * params->npages_per_block));
|
|
start_block++;
|
|
base_address = params->nand_base;
|
|
mini_ctrl_cmd |= GCMD_TWB_VALUE;
|
|
mini_ctrl_cmd |= ERASE_ADDR_SIZE;
|
|
mini_ctrl_cmd |= ((gen_row_address) & (U32_MASK_VAL));
|
|
mini_ctrl_cmd |= PAGE_ERASE_CMD;
|
|
ret = cdns_generic_send_cmd(params, mini_ctrl_cmd);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
NAND_INT_SEM_TAKE(params);
|
|
}
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* Read data from the Cadence NAND controller into a buffer.
|
|
*/
|
|
static inline int cdns_read_data(struct cadence_nand_params *params, uint32_t start_page_number,
|
|
const void *buffer, uint32_t page_count)
|
|
{
|
|
int ret;
|
|
|
|
#if CONFIG_CDNS_NAND_CDMA_MODE
|
|
ret = cdns_nand_cdma_read(params, start_page_number, (char *)buffer, page_count);
|
|
#elif CONFIG_CDNS_NAND_PIO_MODE
|
|
ret = cdns_nand_pio_read_write(params, start_page_number, (char *)buffer, page_count,
|
|
CDNS_READ);
|
|
#elif CONFIG_CDNS_NAND_GENERIC_MODE
|
|
ret = cdns_nand_gen_read_write(params, start_page_number, (char *)buffer, page_count,
|
|
CDNS_READ);
|
|
#endif
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* Read data from the Cadence NAND controller into a buffer.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param buffer The buffer to store the read data.
|
|
* @param offset The offset within the NAND to start reading from.
|
|
* @param size The size of the data to read.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
int cdns_nand_read(struct cadence_nand_params *params, const void *buffer, uint32_t offset,
|
|
uint32_t size)
|
|
{
|
|
uint32_t start_page_number;
|
|
uint32_t end_page_number;
|
|
uint32_t page_count;
|
|
int ret = 0;
|
|
uint16_t r_bytes;
|
|
uint16_t bytes_dif;
|
|
uint16_t lp_bytes_dif;
|
|
uint8_t check_page_first = 0;
|
|
uint8_t check_page_last = 0;
|
|
uint8_t *first_end_page;
|
|
uint8_t *last_end_page;
|
|
|
|
if (params == NULL) {
|
|
LOG_ERR("Wrong parameter passed!!");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (size == 0) {
|
|
return 0;
|
|
}
|
|
|
|
if ((offset >= params->device_size) || (size > (params->device_size - offset))) {
|
|
LOG_ERR("Wrong offset or size value passed!!");
|
|
return -EINVAL;
|
|
}
|
|
|
|
start_page_number = offset / (params->page_size);
|
|
end_page_number = ((offset + size) - 1) / ((params->page_size));
|
|
|
|
if ((offset % params->page_size) == 0) {
|
|
check_page_first = 1;
|
|
}
|
|
if (((offset + size) % params->page_size) == 0) {
|
|
check_page_last = 1;
|
|
}
|
|
page_count = end_page_number - start_page_number;
|
|
page_count++;
|
|
if ((check_page_last == 1) && (check_page_first == 1)) {
|
|
ret = cdns_read_data(params, start_page_number, (char *)buffer, page_count);
|
|
if (ret != 0) {
|
|
return ret;
|
|
}
|
|
|
|
} else if (((check_page_last == 0) && (check_page_first == 1) && (page_count == 1)) ||
|
|
((check_page_last == 0) && (check_page_first == 0) && (page_count == 1)) ||
|
|
((check_page_last == 1) && (check_page_first == 0) && (page_count == 1))) {
|
|
first_end_page = (char *)k_malloc(sizeof(char) * (params->page_size));
|
|
if (first_end_page != NULL) {
|
|
memset(first_end_page, 0xFF, sizeof(char) * (params->page_size));
|
|
} else {
|
|
LOG_ERR("Memory allocation error occurred %s", __func__);
|
|
return -ENOSR;
|
|
}
|
|
ret = cdns_read_data(params, start_page_number, first_end_page, page_count);
|
|
if (ret != 0) {
|
|
k_free(first_end_page);
|
|
return ret;
|
|
}
|
|
memcpy((char *)buffer, first_end_page + (offset % (params->page_size)), size);
|
|
k_free(first_end_page);
|
|
} else if (((check_page_last == 0) && (check_page_first == 1) && (page_count == 2)) ||
|
|
((check_page_last == 0) && (check_page_first == 0) && (page_count == 2)) ||
|
|
((check_page_last == 1) && (check_page_first == 0) && (page_count == 2))) {
|
|
first_end_page = (char *)k_malloc(sizeof(char) * (params->page_size * 2));
|
|
if (first_end_page != NULL) {
|
|
memset(first_end_page, 0xFF, sizeof(char) * (params->page_size * 2));
|
|
} else {
|
|
LOG_ERR("Memory allocation error occurred %s", __func__);
|
|
return -ENOSR;
|
|
}
|
|
ret = cdns_read_data(params, start_page_number, first_end_page, page_count);
|
|
if (ret < 0) {
|
|
k_free(first_end_page);
|
|
return ret;
|
|
}
|
|
memcpy((char *)buffer, first_end_page + (offset % (params->page_size)), size);
|
|
k_free(first_end_page);
|
|
|
|
} else if ((check_page_last == 0) && (check_page_first == 1) && (page_count > 2)) {
|
|
first_end_page = (char *)k_malloc(sizeof(char) * (params->page_size));
|
|
if (first_end_page != NULL) {
|
|
memset(first_end_page, 0xFF, sizeof(char) * (params->page_size));
|
|
} else {
|
|
LOG_ERR("Memory allocation error occurred %s", __func__);
|
|
return -ENOSR;
|
|
}
|
|
ret = cdns_read_data(params, end_page_number, first_end_page, 1);
|
|
if (ret < 0) {
|
|
k_free(first_end_page);
|
|
return ret;
|
|
}
|
|
r_bytes = (offset + size) % (params->page_size);
|
|
ret = cdns_read_data(params, start_page_number, (char *)buffer, (--page_count));
|
|
if (ret != 0) {
|
|
k_free(first_end_page);
|
|
return ret;
|
|
}
|
|
|
|
memcpy((char *)buffer + ((page_count - 1) * params->page_size), first_end_page,
|
|
r_bytes);
|
|
k_free(first_end_page);
|
|
|
|
} else if ((check_page_last == 1) && (check_page_first == 0) && (page_count > 2)) {
|
|
first_end_page = (char *)k_malloc(sizeof(char) * (params->page_size));
|
|
if (first_end_page != NULL) {
|
|
memset(first_end_page, 0xFF, sizeof(char) * (params->page_size));
|
|
} else {
|
|
LOG_ERR("Memory allocation error occurred %s", __func__);
|
|
return -ENOSR;
|
|
}
|
|
ret = cdns_read_data(params, start_page_number, first_end_page, 1);
|
|
if (ret < 0) {
|
|
k_free(first_end_page);
|
|
return ret;
|
|
}
|
|
r_bytes = (offset) % (params->page_size);
|
|
bytes_dif = (((start_page_number + 1) * params->page_size) - r_bytes);
|
|
r_bytes = (offset + size) % (params->page_size);
|
|
ret = cdns_read_data(params, (++start_page_number), ((char *)buffer + bytes_dif),
|
|
(--page_count));
|
|
if (ret != 0) {
|
|
k_free(first_end_page);
|
|
return ret;
|
|
}
|
|
memcpy((char *)buffer, first_end_page + r_bytes, bytes_dif);
|
|
k_free(first_end_page);
|
|
} else if ((check_page_last == 0) && (check_page_first == 0) && (page_count > 2)) {
|
|
first_end_page = (char *)k_malloc(sizeof(char) * (params->page_size));
|
|
last_end_page = (char *)k_malloc(sizeof(char) * (params->page_size));
|
|
if ((first_end_page != NULL) && (last_end_page != NULL)) {
|
|
memset(first_end_page, 0xFF, sizeof(char) * (params->page_size));
|
|
memset(last_end_page, 0xFF, sizeof(char) * (params->page_size));
|
|
} else {
|
|
LOG_ERR("Memory allocation error occurred %s", __func__);
|
|
return -ENOSR;
|
|
}
|
|
ret = cdns_read_data(params, start_page_number, first_end_page, 1);
|
|
if (ret != 0) {
|
|
k_free(first_end_page);
|
|
k_free(last_end_page);
|
|
return ret;
|
|
}
|
|
r_bytes = (offset) % (params->page_size);
|
|
bytes_dif = (((start_page_number + 1) * params->page_size) - r_bytes);
|
|
lp_bytes_dif = (offset + size) % (params->page_size);
|
|
ret = cdns_read_data(params, end_page_number, last_end_page, 1);
|
|
if (ret != 0) {
|
|
k_free(last_end_page);
|
|
k_free(first_end_page);
|
|
return ret;
|
|
}
|
|
r_bytes = (offset + size) % (params->page_size);
|
|
ret = cdns_read_data(params, (++start_page_number), ((char *)buffer + bytes_dif),
|
|
(page_count - 2));
|
|
if (ret != 0) {
|
|
k_free(last_end_page);
|
|
k_free(first_end_page);
|
|
return ret;
|
|
}
|
|
memcpy((char *)buffer, first_end_page + r_bytes, bytes_dif);
|
|
memcpy(((char *)buffer + bytes_dif +
|
|
((page_count - 2) * (params->npages_per_block))),
|
|
last_end_page, lp_bytes_dif);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Write data from a buffer to the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param buffer The buffer containing the data to be written.
|
|
* @param offset The offset within the NAND to start writing to.
|
|
* @param len The length of the data to write.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
int cdns_nand_write(struct cadence_nand_params *params, const void *buffer, uint32_t offset,
|
|
uint32_t len)
|
|
{
|
|
uint32_t start_page_number;
|
|
uint32_t end_page_number;
|
|
uint32_t page_count;
|
|
int ret = 0;
|
|
|
|
if (params == NULL) {
|
|
LOG_ERR("Wrong parameter passed!!");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (len == 0) {
|
|
return 0;
|
|
}
|
|
|
|
if ((offset >= params->device_size) || (len > (params->device_size - offset))) {
|
|
LOG_ERR("Wrong offset or len value passed!!");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if ((offset % params->page_size) != 0) {
|
|
LOG_ERR("offset not page aligned!!! Page size = 0x%x", params->page_size);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if ((len % params->page_size) != 0) {
|
|
LOG_ERR("length not page aligned!!! Page size = 0x%x", params->page_size);
|
|
return -EINVAL;
|
|
}
|
|
|
|
start_page_number = offset / (params->page_size);
|
|
end_page_number = ((offset + len) - 1) / ((params->page_size));
|
|
page_count = end_page_number - start_page_number;
|
|
|
|
#if CONFIG_CDNS_NAND_CDMA_MODE
|
|
ret = cdns_nand_cdma_write(params, start_page_number, (char *)buffer, ++page_count);
|
|
#elif CONFIG_CDNS_NAND_PIO_MODE
|
|
ret = cdns_nand_pio_read_write(params, start_page_number, (char *)buffer, ++page_count,
|
|
CDNS_WRITE);
|
|
#elif CONFIG_CDNS_NAND_GENERIC_MODE
|
|
ret = cdns_nand_gen_read_write(params, start_page_number, (char *)buffer, ++page_count,
|
|
CDNS_WRITE);
|
|
#endif
|
|
if (ret != 0) {
|
|
LOG_ERR("Cadence driver write Failed!!!");
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* Perform an erase operation on the Cadence NAND controller.
|
|
*
|
|
* @param params The Cadence NAND parameters structure.
|
|
* @param offset The offset within the NAND to start erasing.
|
|
* @param size The size of the data to erase.
|
|
* @retval 0 on success or negative error value on failure.
|
|
*/
|
|
int cdns_nand_erase(struct cadence_nand_params *params, uint32_t offset, uint32_t size)
|
|
{
|
|
uint32_t start_block_number;
|
|
uint32_t end_block_number;
|
|
uint32_t block_count;
|
|
int ret;
|
|
|
|
if (params == NULL) {
|
|
LOG_ERR("Wrong parameter passed!!");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (size == 0) {
|
|
return 0;
|
|
}
|
|
|
|
if ((offset >= params->device_size) || (size > (params->device_size - offset))) {
|
|
LOG_ERR("Wrong offset or size value passed!!");
|
|
return -EINVAL;
|
|
}
|
|
if ((offset % (params->block_size)) != 0) {
|
|
LOG_ERR("Offset value not aligned with block size!! Erase block size = %x",
|
|
params->block_size);
|
|
return -EINVAL;
|
|
}
|
|
if ((size % (params->block_size)) != 0) {
|
|
LOG_ERR("Length value not aligned with block size!! Erase block size = %x",
|
|
params->block_size);
|
|
return -EINVAL;
|
|
}
|
|
|
|
start_block_number = (offset / ((params->page_size))) / (params->npages_per_block);
|
|
end_block_number =
|
|
(((offset + size) - 1) / ((params->page_size))) / (params->npages_per_block);
|
|
block_count = end_block_number - start_block_number;
|
|
#if CONFIG_CDNS_NAND_CDMA_MODE
|
|
ret = cdns_nand_cdma_erase(params, start_block_number, ++block_count);
|
|
#elif CONFIG_CDNS_NAND_PIO_MODE
|
|
ret = cdns_nand_pio_erase(params, NF_TDEF_TRD_NUM, NF_TDEF_DEV_NUM, start_block_number,
|
|
CNF_CMD_ERASE, ++block_count);
|
|
#elif CONFIG_CDNS_NAND_GENERIC_MODE
|
|
ret = cdns_nand_gen_erase(params, start_block_number, ++block_count);
|
|
#endif
|
|
if (ret != 0) {
|
|
LOG_ERR("Cadence driver Erase Failed!!!");
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
#if CONFIG_CDNS_NAND_INTERRUPT_SUPPORT
|
|
void cdns_nand_irq_handler_ll(struct cadence_nand_params *params)
|
|
{
|
|
uint32_t status = 0;
|
|
uint8_t thread_num = 0;
|
|
|
|
status = sys_read32(params->nand_base + THREAD_INTERRUPT_STATUS);
|
|
thread_num = find_lsb_set(status);
|
|
|
|
if (GET_INIT_SET_CHECK(status, (thread_num - 1)) != 0) {
|
|
/* Clear the interrupt*/
|
|
sys_write32(BIT((thread_num - 1)), params->nand_base + THREAD_INTERRUPT_STATUS);
|
|
}
|
|
}
|
|
#endif
|