提交 4e2f6c52 编写于 作者: A Abhishek Sahu 提交者: Boris Brezillon

mtd: nand: qcom: QPIC data descriptors handling

1. Add the data descriptor preparation function which will be used
   only by BAM DMA for forming the data SGL’s
2. Add clear BAM transaction and call it before every new request
3. Check DMA mode for ADM or BAM and call the appropriate
   descriptor formation function.
Reviewed-by: NArchit Taneja <architt@codeaurora.org>
Signed-off-by: NAbhishek Sahu <absahu@codeaurora.org>
Signed-off-by: NBoris Brezillon <boris.brezillon@free-electrons.com>
上级 9d43f915
...@@ -473,6 +473,27 @@ alloc_bam_transaction(struct qcom_nand_controller *nandc) ...@@ -473,6 +473,27 @@ alloc_bam_transaction(struct qcom_nand_controller *nandc)
return bam_txn; return bam_txn;
} }
/* Clears the BAM transaction indexes */
static void clear_bam_transaction(struct qcom_nand_controller *nandc)
{
struct bam_transaction *bam_txn = nandc->bam_txn;
if (!nandc->props->is_bam)
return;
bam_txn->cmd_sgl_pos = 0;
bam_txn->cmd_sgl_start = 0;
bam_txn->tx_sgl_pos = 0;
bam_txn->tx_sgl_start = 0;
bam_txn->rx_sgl_pos = 0;
bam_txn->rx_sgl_start = 0;
sg_init_table(bam_txn->cmd_sgl, nandc->max_cwperpage *
QPIC_PER_CW_CMD_SGL);
sg_init_table(bam_txn->data_sgl, nandc->max_cwperpage *
QPIC_PER_CW_DATA_SGL);
}
static inline struct qcom_nand_host *to_qcom_nand_host(struct nand_chip *chip) static inline struct qcom_nand_host *to_qcom_nand_host(struct nand_chip *chip)
{ {
return container_of(chip, struct qcom_nand_host, chip); return container_of(chip, struct qcom_nand_host, chip);
...@@ -704,6 +725,41 @@ static int prepare_bam_async_desc(struct qcom_nand_controller *nandc, ...@@ -704,6 +725,41 @@ static int prepare_bam_async_desc(struct qcom_nand_controller *nandc,
return 0; return 0;
} }
/*
* Prepares the data descriptor for BAM DMA which will be used for NAND
* data reads and writes.
*/
static int prep_bam_dma_desc_data(struct qcom_nand_controller *nandc, bool read,
const void *vaddr,
int size, unsigned int flags)
{
int ret;
struct bam_transaction *bam_txn = nandc->bam_txn;
if (read) {
sg_set_buf(&bam_txn->data_sgl[bam_txn->rx_sgl_pos],
vaddr, size);
bam_txn->rx_sgl_pos++;
} else {
sg_set_buf(&bam_txn->data_sgl[bam_txn->tx_sgl_pos],
vaddr, size);
bam_txn->tx_sgl_pos++;
/*
* BAM will only set EOT for DMA_PREP_INTERRUPT so if this flag
* is not set, form the DMA descriptor
*/
if (!(flags & NAND_BAM_NO_EOT)) {
ret = prepare_bam_async_desc(nandc, nandc->tx_chan,
DMA_PREP_INTERRUPT);
if (ret)
return ret;
}
}
return 0;
}
static int prep_adm_dma_desc(struct qcom_nand_controller *nandc, bool read, static int prep_adm_dma_desc(struct qcom_nand_controller *nandc, bool read,
int reg_off, const void *vaddr, int size, int reg_off, const void *vaddr, int size,
bool flow_control) bool flow_control)
...@@ -854,6 +910,9 @@ static int write_reg_dma(struct qcom_nand_controller *nandc, int first, ...@@ -854,6 +910,9 @@ static int write_reg_dma(struct qcom_nand_controller *nandc, int first,
static int read_data_dma(struct qcom_nand_controller *nandc, int reg_off, static int read_data_dma(struct qcom_nand_controller *nandc, int reg_off,
const u8 *vaddr, int size, unsigned int flags) const u8 *vaddr, int size, unsigned int flags)
{ {
if (nandc->props->is_bam)
return prep_bam_dma_desc_data(nandc, true, vaddr, size, flags);
return prep_adm_dma_desc(nandc, true, reg_off, vaddr, size, false); return prep_adm_dma_desc(nandc, true, reg_off, vaddr, size, false);
} }
...@@ -869,6 +928,9 @@ static int read_data_dma(struct qcom_nand_controller *nandc, int reg_off, ...@@ -869,6 +928,9 @@ static int read_data_dma(struct qcom_nand_controller *nandc, int reg_off,
static int write_data_dma(struct qcom_nand_controller *nandc, int reg_off, static int write_data_dma(struct qcom_nand_controller *nandc, int reg_off,
const u8 *vaddr, int size, unsigned int flags) const u8 *vaddr, int size, unsigned int flags)
{ {
if (nandc->props->is_bam)
return prep_bam_dma_desc_data(nandc, false, vaddr, size, flags);
return prep_adm_dma_desc(nandc, false, reg_off, vaddr, size, false); return prep_adm_dma_desc(nandc, false, reg_off, vaddr, size, false);
} }
...@@ -1156,6 +1218,10 @@ static void pre_command(struct qcom_nand_host *host, int command) ...@@ -1156,6 +1218,10 @@ static void pre_command(struct qcom_nand_host *host, int command)
host->last_command = command; host->last_command = command;
clear_read_regs(nandc); clear_read_regs(nandc);
if (command == NAND_CMD_RESET || command == NAND_CMD_READID ||
command == NAND_CMD_PARAM || command == NAND_CMD_ERASE1)
clear_bam_transaction(nandc);
} }
/* /*
...@@ -1559,6 +1625,7 @@ static int qcom_nandc_read_page(struct mtd_info *mtd, struct nand_chip *chip, ...@@ -1559,6 +1625,7 @@ static int qcom_nandc_read_page(struct mtd_info *mtd, struct nand_chip *chip,
data_buf = buf; data_buf = buf;
oob_buf = oob_required ? chip->oob_poi : NULL; oob_buf = oob_required ? chip->oob_poi : NULL;
clear_bam_transaction(nandc);
ret = read_page_ecc(host, data_buf, oob_buf); ret = read_page_ecc(host, data_buf, oob_buf);
if (ret) { if (ret) {
dev_err(nandc->dev, "failure to read page\n"); dev_err(nandc->dev, "failure to read page\n");
...@@ -1584,6 +1651,8 @@ static int qcom_nandc_read_page_raw(struct mtd_info *mtd, ...@@ -1584,6 +1651,8 @@ static int qcom_nandc_read_page_raw(struct mtd_info *mtd,
oob_buf = chip->oob_poi; oob_buf = chip->oob_poi;
host->use_ecc = false; host->use_ecc = false;
clear_bam_transaction(nandc);
update_rw_regs(host, ecc->steps, true); update_rw_regs(host, ecc->steps, true);
config_nand_page_read(nandc); config_nand_page_read(nandc);
...@@ -1655,6 +1724,7 @@ static int qcom_nandc_read_oob(struct mtd_info *mtd, struct nand_chip *chip, ...@@ -1655,6 +1724,7 @@ static int qcom_nandc_read_oob(struct mtd_info *mtd, struct nand_chip *chip,
int ret; int ret;
clear_read_regs(nandc); clear_read_regs(nandc);
clear_bam_transaction(nandc);
host->use_ecc = true; host->use_ecc = true;
set_address(host, 0, page); set_address(host, 0, page);
...@@ -1678,6 +1748,7 @@ static int qcom_nandc_write_page(struct mtd_info *mtd, struct nand_chip *chip, ...@@ -1678,6 +1748,7 @@ static int qcom_nandc_write_page(struct mtd_info *mtd, struct nand_chip *chip,
int i, ret; int i, ret;
clear_read_regs(nandc); clear_read_regs(nandc);
clear_bam_transaction(nandc);
data_buf = (u8 *)buf; data_buf = (u8 *)buf;
oob_buf = chip->oob_poi; oob_buf = chip->oob_poi;
...@@ -1743,6 +1814,7 @@ static int qcom_nandc_write_page_raw(struct mtd_info *mtd, ...@@ -1743,6 +1814,7 @@ static int qcom_nandc_write_page_raw(struct mtd_info *mtd,
int i, ret; int i, ret;
clear_read_regs(nandc); clear_read_regs(nandc);
clear_bam_transaction(nandc);
data_buf = (u8 *)buf; data_buf = (u8 *)buf;
oob_buf = chip->oob_poi; oob_buf = chip->oob_poi;
...@@ -1819,11 +1891,13 @@ static int qcom_nandc_write_oob(struct mtd_info *mtd, struct nand_chip *chip, ...@@ -1819,11 +1891,13 @@ static int qcom_nandc_write_oob(struct mtd_info *mtd, struct nand_chip *chip,
host->use_ecc = true; host->use_ecc = true;
clear_bam_transaction(nandc);
ret = copy_last_cw(host, page); ret = copy_last_cw(host, page);
if (ret) if (ret)
return ret; return ret;
clear_read_regs(nandc); clear_read_regs(nandc);
clear_bam_transaction(nandc);
/* calculate the data and oob size for the last codeword/step */ /* calculate the data and oob size for the last codeword/step */
data_size = ecc->size - ((ecc->steps - 1) << 2); data_size = ecc->size - ((ecc->steps - 1) << 2);
...@@ -1876,6 +1950,7 @@ static int qcom_nandc_block_bad(struct mtd_info *mtd, loff_t ofs) ...@@ -1876,6 +1950,7 @@ static int qcom_nandc_block_bad(struct mtd_info *mtd, loff_t ofs)
*/ */
host->use_ecc = false; host->use_ecc = false;
clear_bam_transaction(nandc);
ret = copy_last_cw(host, page); ret = copy_last_cw(host, page);
if (ret) if (ret)
goto err; goto err;
...@@ -1906,6 +1981,7 @@ static int qcom_nandc_block_markbad(struct mtd_info *mtd, loff_t ofs) ...@@ -1906,6 +1981,7 @@ static int qcom_nandc_block_markbad(struct mtd_info *mtd, loff_t ofs)
int page, ret, status = 0; int page, ret, status = 0;
clear_read_regs(nandc); clear_read_regs(nandc);
clear_bam_transaction(nandc);
/* /*
* to mark the BBM as bad, we flash the entire last codeword with 0s. * to mark the BBM as bad, we flash the entire last codeword with 0s.
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册