Lines Matching +full:nand +full:- +full:ecc +full:- +full:strength

1 // SPDX-License-Identifier: GPL-2.0
3 * Support for Macronix external hardware ECC engine for NAND devices, also
10 #include <linux/dma-mapping.h>
18 #include <linux/mtd/nand.h>
19 #include <linux/mtd/nand-ecc-mxic.h>
53 /* ECC Chunk Size */
63 /* ECC Chunk Count */
98 /* ECC machinery */
124 static struct mxic_ecc_engine *nand_to_mxic(struct nand_device *nand) in nand_to_mxic() argument
126 struct nand_ecc_engine *eng = nand->ecc.engine; in nand_to_mxic()
128 if (eng->integration == NAND_ECC_ENGINE_INTEGRATION_EXTERNAL) in nand_to_mxic()
137 struct nand_device *nand = mtd_to_nanddev(mtd); in mxic_ecc_ooblayout_ecc() local
138 struct mxic_ecc_ctx *ctx = nand_to_ecc_ctx(nand); in mxic_ecc_ooblayout_ecc()
140 if (section < 0 || section >= ctx->steps) in mxic_ecc_ooblayout_ecc()
141 return -ERANGE; in mxic_ecc_ooblayout_ecc()
143 oobregion->offset = (section * ctx->oob_step_sz) + ctx->meta_sz; in mxic_ecc_ooblayout_ecc()
144 oobregion->length = ctx->parity_sz; in mxic_ecc_ooblayout_ecc()
152 struct nand_device *nand = mtd_to_nanddev(mtd); in mxic_ecc_ooblayout_free() local
153 struct mxic_ecc_ctx *ctx = nand_to_ecc_ctx(nand); in mxic_ecc_ooblayout_free()
155 if (section < 0 || section >= ctx->steps) in mxic_ecc_ooblayout_free()
156 return -ERANGE; in mxic_ecc_ooblayout_free()
159 oobregion->offset = 2; in mxic_ecc_ooblayout_free()
160 oobregion->length = ctx->meta_sz - 2; in mxic_ecc_ooblayout_free()
162 oobregion->offset = section * ctx->oob_step_sz; in mxic_ecc_ooblayout_free()
163 oobregion->length = ctx->meta_sz; in mxic_ecc_ooblayout_free()
170 .ecc = mxic_ecc_ooblayout_ecc,
178 reg = readl(mxic->regs + DP_CONFIG); in mxic_ecc_disable_engine()
180 writel(reg, mxic->regs + DP_CONFIG); in mxic_ecc_disable_engine()
187 reg = readl(mxic->regs + DP_CONFIG); in mxic_ecc_enable_engine()
189 writel(reg, mxic->regs + DP_CONFIG); in mxic_ecc_enable_engine()
194 writel(0, mxic->regs + INTRPT_SIG_EN); in mxic_ecc_disable_int()
199 writel(TRANS_CMPLT, mxic->regs + INTRPT_SIG_EN); in mxic_ecc_enable_int()
207 sts = readl(mxic->regs + INTRPT_STS); in mxic_ecc_isr()
212 complete(&mxic->complete); in mxic_ecc_isr()
214 writel(sts, mxic->regs + INTRPT_STS); in mxic_ecc_isr()
219 static int mxic_ecc_init_ctx(struct nand_device *nand, struct device *dev) in mxic_ecc_init_ctx() argument
221 struct mxic_ecc_engine *mxic = nand_to_mxic(nand); in mxic_ecc_init_ctx()
222 struct nand_ecc_props *conf = &nand->ecc.ctx.conf; in mxic_ecc_init_ctx()
223 struct nand_ecc_props *reqs = &nand->ecc.requirements; in mxic_ecc_init_ctx()
224 struct nand_ecc_props *user = &nand->ecc.user_conf; in mxic_ecc_init_ctx()
225 struct mtd_info *mtd = nanddev_to_mtd(nand); in mxic_ecc_init_ctx()
226 int step_size = 0, strength = 0, desired_correction = 0, steps, idx; in mxic_ecc_init_ctx() local
235 return -ENOMEM; in mxic_ecc_init_ctx()
237 nand->ecc.ctx.priv = ctx; in mxic_ecc_init_ctx()
239 /* Only large page NAND chips may use BCH */ in mxic_ecc_init_ctx()
240 if (mtd->oobsize < 64) { in mxic_ecc_init_ctx()
241 pr_err("BCH cannot be used with small page NAND chips\n"); in mxic_ecc_init_ctx()
242 return -EINVAL; in mxic_ecc_init_ctx()
249 TO_SPARE | TO_MAIN, mxic->regs + INTRPT_STS_EN); in mxic_ecc_init_ctx()
251 /* Configure the correction depending on the NAND device topology */ in mxic_ecc_init_ctx()
252 if (user->step_size && user->strength) { in mxic_ecc_init_ctx()
253 step_size = user->step_size; in mxic_ecc_init_ctx()
254 strength = user->strength; in mxic_ecc_init_ctx()
255 } else if (reqs->step_size && reqs->strength) { in mxic_ecc_init_ctx()
256 step_size = reqs->step_size; in mxic_ecc_init_ctx()
257 strength = reqs->strength; in mxic_ecc_init_ctx()
260 if (step_size && strength) { in mxic_ecc_init_ctx()
261 steps = mtd->writesize / step_size; in mxic_ecc_init_ctx()
262 desired_correction = steps * strength; in mxic_ecc_init_ctx()
265 /* Step size is fixed to 1kiB, strength may vary (4 possible values) */ in mxic_ecc_init_ctx()
266 conf->step_size = SZ_1K; in mxic_ecc_init_ctx()
267 steps = mtd->writesize / conf->step_size; in mxic_ecc_init_ctx()
269 ctx->status = devm_kzalloc(dev, steps * sizeof(u8), GFP_KERNEL); in mxic_ecc_init_ctx()
270 if (!ctx->status) in mxic_ecc_init_ctx()
271 return -ENOMEM; in mxic_ecc_init_ctx()
274 strength = desired_correction / steps; in mxic_ecc_init_ctx()
277 if (possible_strength[idx] >= strength) in mxic_ecc_init_ctx()
281 ARRAY_SIZE(possible_strength) - 1); in mxic_ecc_init_ctx()
284 idx = ARRAY_SIZE(possible_strength) - 1; in mxic_ecc_init_ctx()
287 /* Tune the selected strength until it fits in the OOB area */ in mxic_ecc_init_ctx()
288 for (; idx >= 0; idx--) { in mxic_ecc_init_ctx()
289 if (spare_size[idx] * steps <= mtd->oobsize) in mxic_ecc_init_ctx()
293 /* This engine cannot be used with this NAND device */ in mxic_ecc_init_ctx()
295 return -EINVAL; in mxic_ecc_init_ctx()
297 /* Configure the engine for the desired strength */ in mxic_ecc_init_ctx()
298 writel(ECC_TYP(idx), mxic->regs + DP_CONFIG); in mxic_ecc_init_ctx()
299 conf->strength = possible_strength[idx]; in mxic_ecc_init_ctx()
300 spare_reg = readl(mxic->regs + SPARE_SIZE); in mxic_ecc_init_ctx()
302 ctx->steps = steps; in mxic_ecc_init_ctx()
303 ctx->data_step_sz = mtd->writesize / steps; in mxic_ecc_init_ctx()
304 ctx->oob_step_sz = mtd->oobsize / steps; in mxic_ecc_init_ctx()
305 ctx->parity_sz = PARITY_SZ(spare_reg); in mxic_ecc_init_ctx()
306 ctx->meta_sz = META_SZ(spare_reg); in mxic_ecc_init_ctx()
309 ctx->req_ctx.oob_buffer_size = nanddev_per_page_oobsize(nand) + in mxic_ecc_init_ctx()
310 (ctx->steps * STAT_BYTES); in mxic_ecc_init_ctx()
311 ret = nand_ecc_init_req_tweaking(&ctx->req_ctx, nand); in mxic_ecc_init_ctx()
315 ctx->oobwithstat = kmalloc(mtd->oobsize + (ctx->steps * STAT_BYTES), in mxic_ecc_init_ctx()
317 if (!ctx->oobwithstat) { in mxic_ecc_init_ctx()
318 ret = -ENOMEM; in mxic_ecc_init_ctx()
322 sg_init_table(ctx->sg, 2); in mxic_ecc_init_ctx()
326 readl(mxic->regs + DP_VER) >> DP_VER_OFFSET); in mxic_ecc_init_ctx()
327 dev_err(dev, "Chunk size: %d\n", readl(mxic->regs + CHUNK_SIZE)); in mxic_ecc_init_ctx()
328 dev_err(dev, "Main size: %d\n", readl(mxic->regs + MAIN_SIZE)); in mxic_ecc_init_ctx()
331 dev_err(dev, "Parity size: %d\n", ctx->parity_sz); in mxic_ecc_init_ctx()
332 dev_err(dev, "Meta size: %d\n", ctx->meta_sz); in mxic_ecc_init_ctx()
334 if ((ctx->meta_sz + ctx->parity_sz + RSV_SZ(spare_reg)) != in mxic_ecc_init_ctx()
337 ctx->meta_sz, ctx->parity_sz, RSV_SZ(spare_reg), in mxic_ecc_init_ctx()
339 ret = -EINVAL; in mxic_ecc_init_ctx()
343 if (ctx->oob_step_sz != SPARE_SZ(spare_reg)) { in mxic_ecc_init_ctx()
345 ctx->oob_step_sz, SPARE_SZ(spare_reg)); in mxic_ecc_init_ctx()
346 ret = -EINVAL; in mxic_ecc_init_ctx()
353 kfree(ctx->oobwithstat); in mxic_ecc_init_ctx()
355 nand_ecc_cleanup_req_tweaking(&ctx->req_ctx); in mxic_ecc_init_ctx()
360 static int mxic_ecc_init_ctx_external(struct nand_device *nand) in mxic_ecc_init_ctx_external() argument
362 struct mxic_ecc_engine *mxic = nand_to_mxic(nand); in mxic_ecc_init_ctx_external()
363 struct device *dev = nand->ecc.engine->dev; in mxic_ecc_init_ctx_external()
366 dev_info(dev, "Macronix ECC engine in external mode\n"); in mxic_ecc_init_ctx_external()
368 ret = mxic_ecc_init_ctx(nand, dev); in mxic_ecc_init_ctx_external()
373 writel(1, mxic->regs + CHUNK_CNT); in mxic_ecc_init_ctx_external()
375 mxic->regs + HC_CONFIG); in mxic_ecc_init_ctx_external()
380 static int mxic_ecc_init_ctx_pipelined(struct nand_device *nand) in mxic_ecc_init_ctx_pipelined() argument
382 struct mxic_ecc_engine *mxic = nand_to_mxic(nand); in mxic_ecc_init_ctx_pipelined()
387 dev = nand_ecc_get_engine_dev(nand->ecc.engine->dev); in mxic_ecc_init_ctx_pipelined()
389 return -EINVAL; in mxic_ecc_init_ctx_pipelined()
391 dev_info(dev, "Macronix ECC engine in pipelined/mapping mode\n"); in mxic_ecc_init_ctx_pipelined()
393 ret = mxic_ecc_init_ctx(nand, dev); in mxic_ecc_init_ctx_pipelined()
397 ctx = nand_to_ecc_ctx(nand); in mxic_ecc_init_ctx_pipelined()
400 writel(ctx->steps, mxic->regs + CHUNK_CNT); in mxic_ecc_init_ctx_pipelined()
403 * Interleaved ECC scheme cannot be used otherwise factory bad block in mxic_ecc_init_ctx_pipelined()
407 mxic->regs + HC_CONFIG); in mxic_ecc_init_ctx_pipelined()
412 static void mxic_ecc_cleanup_ctx(struct nand_device *nand) in mxic_ecc_cleanup_ctx() argument
414 struct mxic_ecc_ctx *ctx = nand_to_ecc_ctx(nand); in mxic_ecc_cleanup_ctx()
417 nand_ecc_cleanup_req_tweaking(&ctx->req_ctx); in mxic_ecc_cleanup_ctx()
418 kfree(ctx->oobwithstat); in mxic_ecc_cleanup_ctx()
427 if (mxic->irq) { in mxic_ecc_data_xfer_wait_for_completion()
428 reinit_completion(&mxic->complete); in mxic_ecc_data_xfer_wait_for_completion()
430 ret = wait_for_completion_timeout(&mxic->complete, in mxic_ecc_data_xfer_wait_for_completion()
434 ret = readl_poll_timeout(mxic->regs + INTRPT_STS, val, in mxic_ecc_data_xfer_wait_for_completion()
436 writel(val, mxic->regs + INTRPT_STS); in mxic_ecc_data_xfer_wait_for_completion()
440 dev_err(mxic->dev, "Timeout on data xfer completion\n"); in mxic_ecc_data_xfer_wait_for_completion()
441 return -ETIMEDOUT; in mxic_ecc_data_xfer_wait_for_completion()
457 writel(SDMA_STRT | dir, mxic->regs + SDMA_CTRL); in mxic_ecc_process_data()
473 writel(dirmap, mxic->regs + HC_SLV_ADDR); in mxic_ecc_process_data_pipelined()
481 u8 *buf = ctx->oobwithstat; in mxic_ecc_extract_status_bytes()
485 /* Extract the ECC status */ in mxic_ecc_extract_status_bytes()
486 for (step = 0; step < ctx->steps; step++) { in mxic_ecc_extract_status_bytes()
487 next_stat_pos = ctx->oob_step_sz + in mxic_ecc_extract_status_bytes()
488 ((STAT_BYTES + ctx->oob_step_sz) * step); in mxic_ecc_extract_status_bytes()
490 ctx->status[step] = buf[next_stat_pos]; in mxic_ecc_extract_status_bytes()
499 /* Reconstruct the OOB buffer linearly (without the ECC status bytes) */ in mxic_ecc_reconstruct_oobbuf()
500 for (step = 0; step < ctx->steps; step++) in mxic_ecc_reconstruct_oobbuf()
501 memcpy(dst + (step * ctx->oob_step_sz), in mxic_ecc_reconstruct_oobbuf()
502 src + (step * (ctx->oob_step_sz + STAT_BYTES)), in mxic_ecc_reconstruct_oobbuf()
503 ctx->oob_step_sz); in mxic_ecc_reconstruct_oobbuf()
512 for (step = 0; step < ctx->steps; step++) in mxic_ecc_add_room_in_oobbuf()
513 memcpy(dst + (step * (ctx->oob_step_sz + STAT_BYTES)), in mxic_ecc_add_room_in_oobbuf()
514 src + (step * ctx->oob_step_sz), in mxic_ecc_add_room_in_oobbuf()
515 ctx->oob_step_sz); in mxic_ecc_add_room_in_oobbuf()
519 struct nand_device *nand) in mxic_ecc_count_biterrs() argument
521 struct mxic_ecc_ctx *ctx = nand_to_ecc_ctx(nand); in mxic_ecc_count_biterrs()
522 struct mtd_info *mtd = nanddev_to_mtd(nand); in mxic_ecc_count_biterrs()
523 struct device *dev = mxic->dev; in mxic_ecc_count_biterrs()
528 for (step = 0; step < ctx->steps; step++) { in mxic_ecc_count_biterrs()
529 u8 stat = ctx->status[step]; in mxic_ecc_count_biterrs()
532 dev_dbg(dev, "ECC step %d: no error\n", step); in mxic_ecc_count_biterrs()
534 dev_dbg(dev, "ECC step %d: erased\n", step); in mxic_ecc_count_biterrs()
536 dev_dbg(dev, "ECC step %d: uncorrectable\n", step); in mxic_ecc_count_biterrs()
537 mtd->ecc_stats.failed++; in mxic_ecc_count_biterrs()
540 dev_dbg(dev, "ECC step %d: %d bits corrected\n", in mxic_ecc_count_biterrs()
543 mtd->ecc_stats.corrected += stat; in mxic_ecc_count_biterrs()
547 return failure ? -EBADMSG : max_bf; in mxic_ecc_count_biterrs()
550 /* External ECC engine helpers */
551 static int mxic_ecc_prepare_io_req_external(struct nand_device *nand, in mxic_ecc_prepare_io_req_external() argument
554 struct mxic_ecc_engine *mxic = nand_to_mxic(nand); in mxic_ecc_prepare_io_req_external()
555 struct mxic_ecc_ctx *ctx = nand_to_ecc_ctx(nand); in mxic_ecc_prepare_io_req_external()
556 struct mtd_info *mtd = nanddev_to_mtd(nand); in mxic_ecc_prepare_io_req_external()
559 if (req->mode == MTD_OPS_RAW) in mxic_ecc_prepare_io_req_external()
562 nand_ecc_tweak_req(&ctx->req_ctx, req); in mxic_ecc_prepare_io_req_external()
563 ctx->req = req; in mxic_ecc_prepare_io_req_external()
565 if (req->type == NAND_PAGE_READ) in mxic_ecc_prepare_io_req_external()
568 mxic_ecc_add_room_in_oobbuf(ctx, ctx->oobwithstat, in mxic_ecc_prepare_io_req_external()
569 ctx->req->oobbuf.out); in mxic_ecc_prepare_io_req_external()
571 sg_set_buf(&ctx->sg[0], req->databuf.out, req->datalen); in mxic_ecc_prepare_io_req_external()
572 sg_set_buf(&ctx->sg[1], ctx->oobwithstat, in mxic_ecc_prepare_io_req_external()
573 req->ooblen + (ctx->steps * STAT_BYTES)); in mxic_ecc_prepare_io_req_external()
575 nents = dma_map_sg(mxic->dev, ctx->sg, 2, DMA_BIDIRECTIONAL); in mxic_ecc_prepare_io_req_external()
577 return -EINVAL; in mxic_ecc_prepare_io_req_external()
579 mutex_lock(&mxic->lock); in mxic_ecc_prepare_io_req_external()
581 for (step = 0; step < ctx->steps; step++) { in mxic_ecc_prepare_io_req_external()
582 writel(sg_dma_address(&ctx->sg[0]) + (step * ctx->data_step_sz), in mxic_ecc_prepare_io_req_external()
583 mxic->regs + SDMA_MAIN_ADDR); in mxic_ecc_prepare_io_req_external()
584 writel(sg_dma_address(&ctx->sg[1]) + (step * (ctx->oob_step_sz + STAT_BYTES)), in mxic_ecc_prepare_io_req_external()
585 mxic->regs + SDMA_SPARE_ADDR); in mxic_ecc_prepare_io_req_external()
586 ret = mxic_ecc_process_data(mxic, ctx->req->type); in mxic_ecc_prepare_io_req_external()
591 mutex_unlock(&mxic->lock); in mxic_ecc_prepare_io_req_external()
593 dma_unmap_sg(mxic->dev, ctx->sg, 2, DMA_BIDIRECTIONAL); in mxic_ecc_prepare_io_req_external()
598 /* Retrieve the calculated ECC bytes */ in mxic_ecc_prepare_io_req_external()
599 for (step = 0; step < ctx->steps; step++) { in mxic_ecc_prepare_io_req_external()
600 offset = ctx->meta_sz + (step * ctx->oob_step_sz); in mxic_ecc_prepare_io_req_external()
602 (u8 *)ctx->req->oobbuf.out + offset, in mxic_ecc_prepare_io_req_external()
603 ctx->oobwithstat + (step * STAT_BYTES), in mxic_ecc_prepare_io_req_external()
604 step * ctx->parity_sz, in mxic_ecc_prepare_io_req_external()
605 ctx->parity_sz); in mxic_ecc_prepare_io_req_external()
611 static int mxic_ecc_finish_io_req_external(struct nand_device *nand, in mxic_ecc_finish_io_req_external() argument
614 struct mxic_ecc_engine *mxic = nand_to_mxic(nand); in mxic_ecc_finish_io_req_external()
615 struct mxic_ecc_ctx *ctx = nand_to_ecc_ctx(nand); in mxic_ecc_finish_io_req_external()
618 if (req->mode == MTD_OPS_RAW) in mxic_ecc_finish_io_req_external()
621 if (req->type == NAND_PAGE_WRITE) { in mxic_ecc_finish_io_req_external()
622 nand_ecc_restore_req(&ctx->req_ctx, req); in mxic_ecc_finish_io_req_external()
626 /* Copy the OOB buffer and add room for the ECC engine status bytes */ in mxic_ecc_finish_io_req_external()
627 mxic_ecc_add_room_in_oobbuf(ctx, ctx->oobwithstat, ctx->req->oobbuf.in); in mxic_ecc_finish_io_req_external()
629 sg_set_buf(&ctx->sg[0], req->databuf.in, req->datalen); in mxic_ecc_finish_io_req_external()
630 sg_set_buf(&ctx->sg[1], ctx->oobwithstat, in mxic_ecc_finish_io_req_external()
631 req->ooblen + (ctx->steps * STAT_BYTES)); in mxic_ecc_finish_io_req_external()
632 nents = dma_map_sg(mxic->dev, ctx->sg, 2, DMA_BIDIRECTIONAL); in mxic_ecc_finish_io_req_external()
634 return -EINVAL; in mxic_ecc_finish_io_req_external()
636 mutex_lock(&mxic->lock); in mxic_ecc_finish_io_req_external()
638 for (step = 0; step < ctx->steps; step++) { in mxic_ecc_finish_io_req_external()
639 writel(sg_dma_address(&ctx->sg[0]) + (step * ctx->data_step_sz), in mxic_ecc_finish_io_req_external()
640 mxic->regs + SDMA_MAIN_ADDR); in mxic_ecc_finish_io_req_external()
641 writel(sg_dma_address(&ctx->sg[1]) + (step * (ctx->oob_step_sz + STAT_BYTES)), in mxic_ecc_finish_io_req_external()
642 mxic->regs + SDMA_SPARE_ADDR); in mxic_ecc_finish_io_req_external()
643 ret = mxic_ecc_process_data(mxic, ctx->req->type); in mxic_ecc_finish_io_req_external()
648 mutex_unlock(&mxic->lock); in mxic_ecc_finish_io_req_external()
650 dma_unmap_sg(mxic->dev, ctx->sg, 2, DMA_BIDIRECTIONAL); in mxic_ecc_finish_io_req_external()
653 nand_ecc_restore_req(&ctx->req_ctx, req); in mxic_ecc_finish_io_req_external()
659 mxic_ecc_reconstruct_oobbuf(ctx, ctx->req->oobbuf.in, ctx->oobwithstat); in mxic_ecc_finish_io_req_external()
661 nand_ecc_restore_req(&ctx->req_ctx, req); in mxic_ecc_finish_io_req_external()
663 return mxic_ecc_count_biterrs(mxic, nand); in mxic_ecc_finish_io_req_external()
666 /* Pipelined ECC engine helpers */
667 static int mxic_ecc_prepare_io_req_pipelined(struct nand_device *nand, in mxic_ecc_prepare_io_req_pipelined() argument
670 struct mxic_ecc_engine *mxic = nand_to_mxic(nand); in mxic_ecc_prepare_io_req_pipelined()
671 struct mxic_ecc_ctx *ctx = nand_to_ecc_ctx(nand); in mxic_ecc_prepare_io_req_pipelined()
674 if (req->mode == MTD_OPS_RAW) in mxic_ecc_prepare_io_req_pipelined()
677 nand_ecc_tweak_req(&ctx->req_ctx, req); in mxic_ecc_prepare_io_req_pipelined()
678 ctx->req = req; in mxic_ecc_prepare_io_req_pipelined()
680 /* Copy the OOB buffer and add room for the ECC engine status bytes */ in mxic_ecc_prepare_io_req_pipelined()
681 mxic_ecc_add_room_in_oobbuf(ctx, ctx->oobwithstat, ctx->req->oobbuf.in); in mxic_ecc_prepare_io_req_pipelined()
683 sg_set_buf(&ctx->sg[0], req->databuf.in, req->datalen); in mxic_ecc_prepare_io_req_pipelined()
684 sg_set_buf(&ctx->sg[1], ctx->oobwithstat, in mxic_ecc_prepare_io_req_pipelined()
685 req->ooblen + (ctx->steps * STAT_BYTES)); in mxic_ecc_prepare_io_req_pipelined()
687 nents = dma_map_sg(mxic->dev, ctx->sg, 2, DMA_BIDIRECTIONAL); in mxic_ecc_prepare_io_req_pipelined()
689 return -EINVAL; in mxic_ecc_prepare_io_req_pipelined()
691 mutex_lock(&mxic->lock); in mxic_ecc_prepare_io_req_pipelined()
693 writel(sg_dma_address(&ctx->sg[0]), mxic->regs + SDMA_MAIN_ADDR); in mxic_ecc_prepare_io_req_pipelined()
694 writel(sg_dma_address(&ctx->sg[1]), mxic->regs + SDMA_SPARE_ADDR); in mxic_ecc_prepare_io_req_pipelined()
699 static int mxic_ecc_finish_io_req_pipelined(struct nand_device *nand, in mxic_ecc_finish_io_req_pipelined() argument
702 struct mxic_ecc_engine *mxic = nand_to_mxic(nand); in mxic_ecc_finish_io_req_pipelined()
703 struct mxic_ecc_ctx *ctx = nand_to_ecc_ctx(nand); in mxic_ecc_finish_io_req_pipelined()
706 if (req->mode == MTD_OPS_RAW) in mxic_ecc_finish_io_req_pipelined()
709 mutex_unlock(&mxic->lock); in mxic_ecc_finish_io_req_pipelined()
711 dma_unmap_sg(mxic->dev, ctx->sg, 2, DMA_BIDIRECTIONAL); in mxic_ecc_finish_io_req_pipelined()
713 if (req->type == NAND_PAGE_READ) { in mxic_ecc_finish_io_req_pipelined()
715 mxic_ecc_reconstruct_oobbuf(ctx, ctx->req->oobbuf.in, in mxic_ecc_finish_io_req_pipelined()
716 ctx->oobwithstat); in mxic_ecc_finish_io_req_pipelined()
717 ret = mxic_ecc_count_biterrs(mxic, nand); in mxic_ecc_finish_io_req_pipelined()
720 nand_ecc_restore_req(&ctx->req_ctx, req); in mxic_ecc_finish_io_req_pipelined()
751 /* Retrieve the nand-ecc-engine phandle */ in mxic_ecc_get_pdev()
752 np = of_parse_phandle(spi_pdev->dev.of_node, "nand-ecc-engine", 0); in mxic_ecc_get_pdev()
767 platform_device_put(to_platform_device(mxic->dev)); in mxic_ecc_put_pipelined_engine()
779 return ERR_PTR(-ENODEV); in mxic_ecc_get_pipelined_engine()
784 return ERR_PTR(-EPROBE_DEFER); in mxic_ecc_get_pipelined_engine()
787 return &mxic->pipelined_engine; in mxic_ecc_get_pipelined_engine()
792 * Only the external ECC engine is exported as the pipelined is SoC specific, so
797 struct device *dev = &pdev->dev; in mxic_ecc_probe()
801 mxic = devm_kzalloc(&pdev->dev, sizeof(*mxic), GFP_KERNEL); in mxic_ecc_probe()
803 return -ENOMEM; in mxic_ecc_probe()
805 mxic->dev = &pdev->dev; in mxic_ecc_probe()
808 * Both memory regions for the ECC engine itself and the AXI slave in mxic_ecc_probe()
811 mxic->regs = devm_platform_ioremap_resource(pdev, 0); in mxic_ecc_probe()
812 if (IS_ERR(mxic->regs)) { in mxic_ecc_probe()
813 dev_err(&pdev->dev, "Missing memory region\n"); in mxic_ecc_probe()
814 return PTR_ERR(mxic->regs); in mxic_ecc_probe()
821 mxic->irq = platform_get_irq_byname_optional(pdev, "ecc-engine"); in mxic_ecc_probe()
822 if (mxic->irq > 0) { in mxic_ecc_probe()
823 ret = devm_request_irq(&pdev->dev, mxic->irq, mxic_ecc_isr, 0, in mxic_ecc_probe()
824 "mxic-ecc", mxic); in mxic_ecc_probe()
829 mxic->irq = 0; in mxic_ecc_probe()
832 mutex_init(&mxic->lock); in mxic_ecc_probe()
835 * In external mode, the device is the ECC engine. In pipelined mode, in mxic_ecc_probe()
837 * right ECC engine based on the DT properties. in mxic_ecc_probe()
839 mxic->external_engine.dev = &pdev->dev; in mxic_ecc_probe()
840 mxic->external_engine.integration = NAND_ECC_ENGINE_INTEGRATION_EXTERNAL; in mxic_ecc_probe()
841 mxic->external_engine.ops = &mxic_ecc_engine_external_ops; in mxic_ecc_probe()
843 nand_ecc_register_on_host_hw_engine(&mxic->external_engine); in mxic_ecc_probe()
854 nand_ecc_unregister_on_host_hw_engine(&mxic->external_engine); in mxic_ecc_remove()
861 .compatible = "mxicy,nand-ecc-engine-rev3",
869 .name = "mxic-nand-ecc-engine",
879 MODULE_DESCRIPTION("Macronix NAND hardware ECC controller");