[RFC,v4,2/5] mtd: nand: ecc: mtk: Convert to the ECC infrastructure
Miquel Raynal
miquel.raynal at bootlin.com
Thu Dec 9 02:32:09 PST 2021
Hi Xiangsheng,
xiangsheng.hou at mediatek.com wrote on Tue, 30 Nov 2021 16:31:59 +0800:
> Convert the Mediatek HW ECC engine to the ECC infrastructure with
> pipelined case.
>
> Signed-off-by: Xiangsheng Hou <xiangsheng.hou at mediatek.com>
> ---
> drivers/mtd/nand/ecc-mtk.c | 614 +++++++++++++++++++++++++++++++
> include/linux/mtd/nand-ecc-mtk.h | 68 ++++
> 2 files changed, 682 insertions(+)
>
> diff --git a/drivers/mtd/nand/ecc-mtk.c b/drivers/mtd/nand/ecc-mtk.c
> index 31d7c77d5c59..c44499b3d0a5 100644
> --- a/drivers/mtd/nand/ecc-mtk.c
> +++ b/drivers/mtd/nand/ecc-mtk.c
> @@ -16,6 +16,7 @@
> #include <linux/of_platform.h>
> #include <linux/mutex.h>
>
> +#include <linux/mtd/nand.h>
> #include <linux/mtd/nand-ecc-mtk.h>
>
> #define ECC_IDLE_MASK BIT(0)
> @@ -41,11 +42,17 @@
> #define ECC_IDLE_REG(op) ((op) == ECC_ENCODE ? ECC_ENCIDLE : ECC_DECIDLE)
> #define ECC_CTL_REG(op) ((op) == ECC_ENCODE ? ECC_ENCCON : ECC_DECCON)
>
> +#define OOB_FREE_MAX_SIZE 8
> +#define OOB_FREE_MIN_SIZE 1
> +
> struct mtk_ecc_caps {
> u32 err_mask;
> const u8 *ecc_strength;
> const u32 *ecc_regs;
> u8 num_ecc_strength;
> + const u8 *spare_size;
> + u8 num_spare_size;
> + u32 max_section_size;
> u8 ecc_mode_shift;
> u32 parity_bits;
> int pg_irq_sel;
> @@ -79,6 +86,12 @@ static const u8 ecc_strength_mt7622[] = {
> 4, 6, 8, 10, 12, 14, 16
> };
>
> +/* spare size for each section that each IP supports */
> +static const u8 spare_size_mt7622[] = {
> + 16, 26, 27, 28, 32, 36, 40, 44, 48, 49, 50, 51,
> + 52, 62, 61, 63, 64, 67, 74
> +};
> +
> enum mtk_ecc_regs {
> ECC_ENCPAR00,
> ECC_ENCIRQ_EN,
> @@ -447,6 +460,604 @@ unsigned int mtk_ecc_get_parity_bits(struct mtk_ecc *ecc)
> }
> EXPORT_SYMBOL(mtk_ecc_get_parity_bits);
>
> +static inline int mtk_ecc_data_off(struct nand_device *nand, int i)
> +{
> + int eccsize = nand->ecc.ctx.conf.step_size;
> +
> + return i * eccsize;
> +}
> +
> +static inline int mtk_ecc_oob_free_position(struct nand_device *nand, int i)
> +{
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> + int position;
> +
> + if (i < eng->bbm_ctl.section)
> + position = (i + 1) * eng->oob_free;
> + else if (i == eng->bbm_ctl.section)
> + position = 0;
> + else
> + position = i * eng->oob_free;
> +
> + return position;
> +}
> +
> +static inline int mtk_ecc_data_len(struct nand_device *nand)
> +{
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> + int eccsize = nand->ecc.ctx.conf.step_size;
> + int eccbytes = eng->oob_ecc;
> +
> + return eccsize + eng->oob_free + eccbytes;
> +}
> +
> +static inline u8 *mtk_ecc_section_ptr(struct nand_device *nand, int i)
> +{
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> +
> + return eng->bounce_page_buf + i * mtk_ecc_data_len(nand);
> +}
> +
> +static inline u8 *mtk_ecc_oob_free_ptr(struct nand_device *nand, int i)
> +{
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> + int eccsize = nand->ecc.ctx.conf.step_size;
> +
> + return eng->bounce_page_buf + i * mtk_ecc_data_len(nand) + eccsize;
> +}
> +
> +static void mtk_ecc_no_bbm_swap(struct nand_device *a, u8 *b, u8 *c)
> +{
> + /* nop */
Is this really useful?
> +}
> +
> +static void mtk_ecc_bbm_swap(struct nand_device *nand, u8 *databuf, u8 *oobbuf)
> +{
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> + int step_size = nand->ecc.ctx.conf.step_size;
> + u32 bbm_pos = eng->bbm_ctl.position;
> +
> + bbm_pos += eng->bbm_ctl.section * step_size;
> +
> + swap(oobbuf[0], databuf[bbm_pos]);
> +}
> +
> +static void mtk_ecc_set_bbm_ctl(struct mtk_ecc_bbm_ctl *bbm_ctl,
> + struct nand_device *nand)
> +{
> + if (nanddev_page_size(nand) == 512) {
> + bbm_ctl->bbm_swap = mtk_ecc_no_bbm_swap;
> + } else {
> + bbm_ctl->bbm_swap = mtk_ecc_bbm_swap;
> + bbm_ctl->section = nanddev_page_size(nand) /
> + mtk_ecc_data_len(nand);
> + bbm_ctl->position = nanddev_page_size(nand) %
> + mtk_ecc_data_len(nand);
> + }
> +}
> +
> +static int mtk_ecc_ooblayout_free(struct mtd_info *mtd, int section,
> + struct mtd_oob_region *oob_region)
> +{
> + struct nand_device *nand = mtd_to_nanddev(mtd);
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> + struct nand_ecc_props *conf = &nand->ecc.ctx.conf;
> + u32 eccsteps, bbm_bytes = 0;
> +
> + eccsteps = mtd->writesize / conf->step_size;
> +
> + if (section >= eccsteps)
> + return -ERANGE;
> +
> + /* Reserve 1 byte for BBM only for section 0 */
> + if (section == 0)
> + bbm_bytes = 1;
> +
> + oob_region->length = eng->oob_free - bbm_bytes;
> + oob_region->offset = section * eng->oob_free + bbm_bytes;
> +
> + return 0;
> +}
> +
> +static int mtk_ecc_ooblayout_ecc(struct mtd_info *mtd, int section,
> + struct mtd_oob_region *oob_region)
> +{
> + struct nand_device *nand = mtd_to_nanddev(mtd);
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> +
> + if (section)
> + return -ERANGE;
> +
> + oob_region->offset = eng->oob_free * eng->nsteps;
> + oob_region->length = mtd->oobsize - oob_region->offset;
> +
> + return 0;
> +}
> +
> +static const struct mtd_ooblayout_ops mtk_ecc_ooblayout_ops = {
> + .free = mtk_ecc_ooblayout_free,
> + .ecc = mtk_ecc_ooblayout_ecc,
> +};
> +
> +const struct mtd_ooblayout_ops *mtk_ecc_get_ooblayout(void)
> +{
> + return &mtk_ecc_ooblayout_ops;
> +}
> +
> +static struct device *mtk_ecc_get_engine_dev(struct device *dev)
> +{
> + struct platform_device *eccpdev;
> + struct device_node *np;
> +
> + /*
> + * The device node is only the host controller,
> + * not the actual ECC engine when pipelined case.
> + */
> + np = of_parse_phandle(dev->of_node, "nand-ecc-engine", 0);
> + if (!np)
> + return NULL;
> +
> + eccpdev = of_find_device_by_node(np);
> + if (!eccpdev) {
> + of_node_put(np);
> + return NULL;
> + }
> +
> + platform_device_put(eccpdev);
> + of_node_put(np);
> +
> + return &eccpdev->dev;
> +}
As this will be the exact same function for all the pipelined engines,
I am tempted to put this in the core. I'll soon send a iteration, stay
tuned.
> +/*
> + * mtk_ecc_data_format() - Convert to/from MTK ECC on-flash data format
> + *
> + * MTK ECC engine organize page data by section, the on-flash format as bellow:
> + * || section 0 || section 1 || ...
> + * || data | OOB free | OOB ECC || data || OOB free | OOB ECC || ...
> + *
> + * Terefore, it`s necessary to convert data when reading/writing in raw mode.
> + */
> +static void mtk_ecc_data_format(struct nand_device *nand,
mtk_ecc_reorganize_data_layout()?
> + struct nand_page_io_req *req)
> +{
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> + int step_size = nand->ecc.ctx.conf.step_size;
> + void *databuf, *oobbuf;
> + int i;
> +
> + if (req->type == NAND_PAGE_WRITE) {
> + databuf = (void *)req->databuf.out;
> + oobbuf = (void *)req->oobbuf.out;
> +
> + /*
> + * Convert the source databuf and oobbuf to MTK ECC
> + * on-flash data format.
> + */
> + for (i = 0; i < eng->nsteps; i++) {
> + if (i == eng->bbm_ctl.section)
> + eng->bbm_ctl.bbm_swap(nand,
> + databuf, oobbuf);
Do you really need this swap? Isn't the overall move enough to put the
BBM at the right place?
> + memcpy(mtk_ecc_section_ptr(nand, i),
> + databuf + mtk_ecc_data_off(nand, i),
> + step_size);
> +
> + memcpy(mtk_ecc_oob_free_ptr(nand, i),
> + oobbuf + mtk_ecc_oob_free_position(nand, i),
> + eng->oob_free);
> +
> + memcpy(mtk_ecc_oob_free_ptr(nand, i) + eng->oob_free,
> + oobbuf + eng->oob_free * eng->nsteps +
> + i * eng->oob_ecc,
> + eng->oob_ecc);
> + }
> +
> + req->databuf.out = eng->bounce_page_buf;
> + req->oobbuf.out = eng->bounce_oob_buf;
> + } else {
> + databuf = req->databuf.in;
> + oobbuf = req->oobbuf.in;
> +
> + /*
> + * Convert the on-flash MTK ECC data format to
> + * destination databuf and oobbuf.
> + */
> + memcpy(eng->bounce_page_buf, databuf,
> + nanddev_page_size(nand));
> + memcpy(eng->bounce_oob_buf, oobbuf,
> + nanddev_per_page_oobsize(nand));
> +
> + for (i = 0; i < eng->nsteps; i++) {
> + memcpy(databuf + mtk_ecc_data_off(nand, i),
> + mtk_ecc_section_ptr(nand, i), step_size);
> +
> + memcpy(oobbuf + mtk_ecc_oob_free_position(nand, i),
> + mtk_ecc_section_ptr(nand, i) + step_size,
> + eng->oob_free);
> +
> + memcpy(oobbuf + eng->oob_free * eng->nsteps +
> + i * eng->oob_ecc,
> + mtk_ecc_section_ptr(nand, i) + step_size
> + + eng->oob_free,
> + eng->oob_ecc);
> +
> + if (i == eng->bbm_ctl.section)
> + eng->bbm_ctl.bbm_swap(nand,
> + databuf, oobbuf);
> + }
> + }
> +}
> +
> +static void mtk_ecc_oob_free_shift(struct nand_device *nand,
> + u8 *dst_buf, u8 *src_buf, bool write)
> +{
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> + u32 position;
> + int i;
> +
> + for (i = 0; i < eng->nsteps; i++) {
> + if (i < eng->bbm_ctl.section)
> + position = (i + 1) * eng->oob_free;
> + else if (i == eng->bbm_ctl.section)
> + position = 0;
> + else
> + position = i * eng->oob_free;
> +
> + if (write)
> + memcpy(dst_buf + i * eng->oob_free, src_buf + position,
> + eng->oob_free);
> + else
> + memcpy(dst_buf + position, src_buf + i * eng->oob_free,
> + eng->oob_free);
> + }
> +}
> +
> +static void mtk_ecc_set_section_size_and_strength(struct nand_device *nand)
> +{
> + struct nand_ecc_props *reqs = &nand->ecc.requirements;
> + struct nand_ecc_props *user = &nand->ecc.user_conf;
> + struct nand_ecc_props *conf = &nand->ecc.ctx.conf;
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> +
> + /* Configure the correction depending on the NAND device topology */
> + if (user->step_size && user->strength) {
> + conf->step_size = user->step_size;
> + conf->strength = user->strength;
> + } else if (reqs->step_size && reqs->strength) {
> + conf->step_size = reqs->step_size;
> + conf->strength = reqs->strength;
> + }
> +
> + /*
> + * Align ECC strength and ECC size.
> + * The MTK HW ECC engine only support 512 and 1024 ECC size.
> + */
> + if (conf->step_size < 1024) {
I prefer stronger checks than '<'.
> + if (nanddev_page_size(nand) > 512 &&
> + eng->ecc->caps->max_section_size > 512) {
> + conf->step_size = 1024;
> + conf->strength <<= 1;
the operation "<<= 1" is more readable as "* 2" IMHO.
Same below in both directions.
> + } else {
> + conf->step_size = 512;
> + }
> + } else {
> + conf->step_size = 1024;
> + }
> +
> + eng->section_size = conf->step_size;
> +}
> +
> +static int mtk_ecc_set_spare_per_section(struct nand_device *nand)
> +{
> + struct nand_ecc_props *conf = &nand->ecc.ctx.conf;
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> + const u8 *spare = eng->ecc->caps->spare_size;
> + u32 i, closest_spare = 0;
> +
> + eng->nsteps = nanddev_page_size(nand) / conf->step_size;
> + eng->oob_per_section = nanddev_per_page_oobsize(nand) / eng->nsteps;
> +
> + if (conf->step_size == 1024)
> + eng->oob_per_section >>= 1;
> +
> + if (eng->oob_per_section < spare[0]) {
> + dev_err(eng->ecc->dev, "OOB size per section too small %d\n",
> + eng->oob_per_section);
> + return -EINVAL;
> + }
> +
> + for (i = 0; i < eng->ecc->caps->num_spare_size; i++) {
> + if (eng->oob_per_section >= spare[i] &&
> + spare[i] >= spare[closest_spare]) {
> + closest_spare = i;
> + if (eng->oob_per_section == spare[i])
> + break;
> + }
> + }
> +
> + eng->oob_per_section = spare[closest_spare];
> + eng->oob_per_section_idx = closest_spare;
> +
> + if (conf->step_size == 1024)
> + eng->oob_per_section <<= 1;
> +
> + return 0;
> +}
> +
> +int mtk_ecc_prepare_io_req_pipelined(struct nand_device *nand,
> + struct nand_page_io_req *req)
> +{
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> + struct mtd_info *mtd = nanddev_to_mtd(nand);
> + int ret;
> +
> + nand_ecc_tweak_req(&eng->req_ctx, req);
> +
> + /* Store the source buffer data to avoid modify source data */
> + if (req->type == NAND_PAGE_WRITE) {
> + if (req->datalen)
> + memcpy(eng->src_page_buf + req->dataoffs,
> + req->databuf.out,
> + req->datalen);
> +
> + if (req->ooblen)
> + memcpy(eng->src_oob_buf + req->ooboffs,
> + req->oobbuf.out,
> + req->ooblen);
> + }
> +
> + if (req->mode == MTD_OPS_RAW) {
> + if (req->type == NAND_PAGE_WRITE)
> + mtk_ecc_data_format(nand, req);
> +
> + return 0;
> + }
> +
> + eng->ecc_cfg.mode = ECC_NFI_MODE;
> + eng->ecc_cfg.sectors = eng->nsteps;
> + eng->ecc_cfg.op = ECC_DECODE;
> +
> + if (req->type == NAND_PAGE_READ)
> + return mtk_ecc_enable(eng->ecc, &eng->ecc_cfg);
> +
> + memset(eng->bounce_oob_buf, 0xff, nanddev_per_page_oobsize(nand));
> + if (req->ooblen) {
> + if (req->mode == MTD_OPS_AUTO_OOB) {
> + ret = mtd_ooblayout_set_databytes(mtd,
> + req->oobbuf.out,
> + eng->bounce_oob_buf,
> + req->ooboffs,
> + mtd->oobavail);
> + if (ret)
> + return ret;
> + } else {
> + memcpy(eng->bounce_oob_buf + req->ooboffs,
> + req->oobbuf.out,
> + req->ooblen);
> + }
> + }
> +
> + eng->bbm_ctl.bbm_swap(nand, (void *)req->databuf.out,
> + eng->bounce_oob_buf);
> + mtk_ecc_oob_free_shift(nand, (void *)req->oobbuf.out,
> + eng->bounce_oob_buf, true);
> +
> + eng->ecc_cfg.op = ECC_ENCODE;
> +
> + return mtk_ecc_enable(eng->ecc, &eng->ecc_cfg);
> +}
> +
> +int mtk_ecc_finish_io_req_pipelined(struct nand_device *nand,
> + struct nand_page_io_req *req)
> +{
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> + struct mtd_info *mtd = nanddev_to_mtd(nand);
> + struct mtk_ecc_stats stats;
> + int ret;
> +
> + if (req->type == NAND_PAGE_WRITE) {
> + /* Restore the source buffer data */
> + if (req->datalen)
> + memcpy((void *)req->databuf.out,
> + eng->src_page_buf + req->dataoffs,
> + req->datalen);
> +
> + if (req->ooblen)
> + memcpy((void *)req->oobbuf.out,
> + eng->src_oob_buf + req->ooboffs,
> + req->ooblen);
> +
> + if (req->mode != MTD_OPS_RAW)
> + mtk_ecc_disable(eng->ecc);
> +
> + nand_ecc_restore_req(&eng->req_ctx, req);
> +
> + return 0;
> + }
> +
> + if (req->mode == MTD_OPS_RAW) {
> + mtk_ecc_data_format(nand, req);
> + nand_ecc_restore_req(&eng->req_ctx, req);
> +
> + return 0;
> + }
> +
> + ret = mtk_ecc_wait_done(eng->ecc, ECC_DECODE);
> + if (ret) {
> + ret = -ETIMEDOUT;
> + goto out;
> + }
> +
> + if (eng->read_empty) {
> + memset(req->databuf.in, 0xff, nanddev_page_size(nand));
> + memset(req->oobbuf.in, 0xff, nanddev_per_page_oobsize(nand));
> + ret = 0;
> +
> + goto out;
> + }
> +
> + mtk_ecc_get_stats(eng->ecc, &stats, eng->nsteps);
> + mtd->ecc_stats.corrected += stats.corrected;
> + mtd->ecc_stats.failed += stats.failed;
> +
> + /*
> + * Return -EBADMSG when exit uncorrect ECC error.
> + * Otherwise, return the bitflips.
> + */
> + if (stats.failed)
> + ret = -EBADMSG;
> + else
> + ret = stats.bitflips;
> +
> + memset(eng->bounce_oob_buf, 0xff, nanddev_per_page_oobsize(nand));
> + mtk_ecc_oob_free_shift(nand, eng->bounce_oob_buf, req->oobbuf.in, false);
> + eng->bbm_ctl.bbm_swap(nand, req->databuf.in, eng->bounce_oob_buf);
> +
> + if (req->ooblen) {
> + if (req->mode == MTD_OPS_AUTO_OOB)
> + ret = mtd_ooblayout_get_databytes(mtd,
> + req->oobbuf.in,
> + eng->bounce_oob_buf,
> + req->ooboffs,
> + mtd->oobavail);
> + else
> + memcpy(req->oobbuf.in,
> + eng->bounce_oob_buf + req->ooboffs,
> + req->ooblen);
> + }
> +
> +out:
> + mtk_ecc_disable(eng->ecc);
> + nand_ecc_restore_req(&eng->req_ctx, req);
> +
> + return ret;
> +}
> +
> +int mtk_ecc_init_ctx_pipelined(struct nand_device *nand)
> +{
> + struct nand_ecc_props *conf = &nand->ecc.ctx.conf;
> + struct mtd_info *mtd = nanddev_to_mtd(nand);
> + struct mtk_ecc_engine *eng;
> + struct device *dev;
> + int free, ret;
> +
> + /*
> + * In the case of a pipelined engine, the device registering the ECC
> + * engine is not the actual ECC engine device but the host controller.
> + */
> + dev = mtk_ecc_get_engine_dev(nand->ecc.engine->dev);
> + if (!dev)
> + return -EINVAL;
> +
> + eng = devm_kzalloc(dev, sizeof(*eng), GFP_KERNEL);
> + if (!eng)
> + return -ENOMEM;
> +
> + nand->ecc.ctx.priv = eng;
> + nand->ecc.engine->priv = eng;
> +
> + eng->ecc = dev_get_drvdata(dev);
> +
> + mtk_ecc_set_section_size_and_strength(nand);
> +
> + ret = mtk_ecc_set_spare_per_section(nand);
> + if (ret)
> + return ret;
> +
> + clk_prepare_enable(eng->ecc->clk);
> + mtk_ecc_hw_init(eng->ecc);
> +
> + /* Calculate OOB free bytes except ECC parity data */
> + free = (conf->strength * mtk_ecc_get_parity_bits(eng->ecc)
> + + 7) >> 3;
> + free = eng->oob_per_section - free;
> +
> + /*
> + * Enhance ECC strength if OOB left is bigger than max FDM size
> + * or reduce ECC strength if OOB size is not enough for ECC
> + * parity data.
> + */
> + if (free > OOB_FREE_MAX_SIZE)
> + eng->oob_ecc = eng->oob_per_section - OOB_FREE_MAX_SIZE;
> + else if (free < 0)
> + eng->oob_ecc = eng->oob_per_section - OOB_FREE_MIN_SIZE;
> +
> + /* Calculate and adjust ECC strenth based on OOB ECC bytes */
> + conf->strength = (eng->oob_ecc << 3) /
> + mtk_ecc_get_parity_bits(eng->ecc);
> + mtk_ecc_adjust_strength(eng->ecc, &conf->strength);
> +
> + eng->oob_ecc = DIV_ROUND_UP(conf->strength *
> + mtk_ecc_get_parity_bits(eng->ecc), 8);
> +
> + eng->oob_free = eng->oob_per_section - eng->oob_ecc;
> + if (eng->oob_free > OOB_FREE_MAX_SIZE)
> + eng->oob_free = OOB_FREE_MAX_SIZE;
> +
> + eng->oob_free_protected = OOB_FREE_MIN_SIZE;
> +
> + eng->oob_ecc = eng->oob_per_section - eng->oob_free;
> +
> + if (!mtd->ooblayout)
> + mtd_set_ooblayout(mtd, mtk_ecc_get_ooblayout());
> +
> + ret = nand_ecc_init_req_tweaking(&eng->req_ctx, nand);
> + if (ret)
> + return ret;
> +
> + eng->src_page_buf = kmalloc(nanddev_page_size(nand) +
> + nanddev_per_page_oobsize(nand), GFP_KERNEL);
> + eng->bounce_page_buf = kmalloc(nanddev_page_size(nand) +
> + nanddev_per_page_oobsize(nand), GFP_KERNEL);
> + if (!eng->src_page_buf || !eng->bounce_page_buf) {
> + ret = -ENOMEM;
> + goto cleanup_req_tweak;
> + }
> +
> + eng->src_oob_buf = eng->src_page_buf + nanddev_page_size(nand);
> + eng->bounce_oob_buf = eng->bounce_page_buf + nanddev_page_size(nand);
> +
> + mtk_ecc_set_bbm_ctl(&eng->bbm_ctl, nand);
> + eng->ecc_cfg.strength = conf->strength;
> + eng->ecc_cfg.len = conf->step_size + eng->oob_free_protected;
> + mtd->bitflip_threshold = conf->strength;
> +
> + return 0;
> +
> +cleanup_req_tweak:
> + nand_ecc_cleanup_req_tweaking(&eng->req_ctx);
> +
> + return ret;
> +}
> +
> +void mtk_ecc_cleanup_ctx_pipelined(struct nand_device *nand)
> +{
> + struct mtk_ecc_engine *eng = nand_to_ecc_ctx(nand);
> +
> + if (eng) {
> + nand_ecc_cleanup_req_tweaking(&eng->req_ctx);
> + kfree(eng->src_page_buf);
> + kfree(eng->bounce_page_buf);
> + }
> +}
> +
> +/*
> + * The MTK ECC engine work at pipelined situation,
> + * will be registered by the drivers that wrap it.
> + */
> +static struct nand_ecc_engine_ops mtk_ecc_engine_pipelined_ops = {
> + .init_ctx = mtk_ecc_init_ctx_pipelined,
> + .cleanup_ctx = mtk_ecc_cleanup_ctx_pipelined,
> + .prepare_io_req = mtk_ecc_prepare_io_req_pipelined,
> + .finish_io_req = mtk_ecc_finish_io_req_pipelined,
> +};
> +
> +struct nand_ecc_engine_ops *mtk_ecc_get_pipelined_ops(void)
> +{
> + return &mtk_ecc_engine_pipelined_ops;
> +}
> +EXPORT_SYMBOL(mtk_ecc_get_pipelined_ops);
> +
> static const struct mtk_ecc_caps mtk_ecc_caps_mt2701 = {
> .err_mask = 0x3f,
> .ecc_strength = ecc_strength_mt2701,
> @@ -472,6 +1083,9 @@ static const struct mtk_ecc_caps mtk_ecc_caps_mt7622 = {
> .ecc_strength = ecc_strength_mt7622,
> .ecc_regs = mt7622_ecc_regs,
> .num_ecc_strength = 7,
> + .spare_size = spare_size_mt7622,
> + .num_spare_size = 19,
> + .max_section_size = 1024,
> .ecc_mode_shift = 4,
> .parity_bits = 13,
> .pg_irq_sel = 0,
> diff --git a/include/linux/mtd/nand-ecc-mtk.h b/include/linux/mtd/nand-ecc-mtk.h
> index 0e48c36e6ca0..6d550032cbd9 100644
> --- a/include/linux/mtd/nand-ecc-mtk.h
> +++ b/include/linux/mtd/nand-ecc-mtk.h
> @@ -33,6 +33,61 @@ struct mtk_ecc_config {
> u32 len;
> };
>
> +/**
> + * struct mtk_ecc_bbm_ctl - Information relative to the BBM swap
> + * @bbm_swap: BBM swap function
> + * @section: Section number in data area for swap
> + * @position: Position in @section for swap with BBM
> + */
> +struct mtk_ecc_bbm_ctl {
> + void (*bbm_swap)(struct nand_device *nand, u8 *databuf, u8 *oobbuf);
> + u32 section;
> + u32 position;
> +};
> +
> +/**
> + * struct mtk_ecc_engine - Information relative to the ECC
> + * @req_ctx: Save request context and tweak the original request to fit the
> + * engine needs
> + * @oob_per_section: OOB size for each section to store OOB free/ECC bytes
> + * @oob_per_section_idx: The index for @oob_per_section in spare size array
> + * @oob_ecc: OOB size for each section to store the ECC parity
> + * @oob_free: OOB size for each section to store the OOB free bytes
> + * @oob_free_protected: OOB free bytes will be protected by the ECC engine
> + * @section_size: The size of each section
> + * @read_empty: Indicate whether empty page for one read operation
> + * @nsteps: The number of the sections
> + * @src_page_buf: Buffer used to store source data buffer when write
> + * @src_oob_buf: Buffer used to store source OOB buffer when write
> + * @bounce_page_buf: Data bounce buffer
> + * @bounce_oob_buf: OOB bounce buffer
> + * @ecc: The ECC engine private data structure
> + * @ecc_cfg: The configuration of each ECC operation
> + * @bbm_ctl: Information relative to the BBM swap
> + */
> +struct mtk_ecc_engine {
> + struct nand_ecc_req_tweak_ctx req_ctx;
> +
> + u32 oob_per_section;
> + u32 oob_per_section_idx;
> + u32 oob_ecc;
> + u32 oob_free;
> + u32 oob_free_protected;
> + u32 section_size;
> +
> + bool read_empty;
> + u32 nsteps;
> +
> + u8 *src_page_buf;
> + u8 *src_oob_buf;
> + u8 *bounce_page_buf;
> + u8 *bounce_oob_buf;
> +
> + struct mtk_ecc *ecc;
> + struct mtk_ecc_config ecc_cfg;
> + struct mtk_ecc_bbm_ctl bbm_ctl;
> +};
This and above should not be exported and be located in the driver.
> +
> int mtk_ecc_encode(struct mtk_ecc *, struct mtk_ecc_config *, u8 *, u32);
> void mtk_ecc_get_stats(struct mtk_ecc *, struct mtk_ecc_stats *, int);
> int mtk_ecc_wait_done(struct mtk_ecc *, enum mtk_ecc_operation);
> @@ -44,4 +99,17 @@ unsigned int mtk_ecc_get_parity_bits(struct mtk_ecc *ecc);
> struct mtk_ecc *of_mtk_ecc_get(struct device_node *);
> void mtk_ecc_release(struct mtk_ecc *);
>
> +#if IS_ENABLED(CONFIG_MTD_NAND_ECC_MTK)
> +
> +struct nand_ecc_engine_ops *mtk_ecc_get_pipelined_ops(void);
> +
> +#else /* !CONFIG_MTD_NAND_ECC_MTK */
> +
> +struct nand_ecc_engine_ops *mtk_ecc_get_pipelined_ops(void)
> +{
> + return NULL;
> +}
> +
> +#endif /* CONFIG_MTD_NAND_ECC_MTK */
> +
> #endif
Thanks,
Miquèl
More information about the Linux-mediatek
mailing list