Hi,

Due the DMA api changes, please ignore this patch.


thanks

Huang Shijie
> [1] Background :
>     The GPMI does ECC read page operation with a DMA chain consist of three 
> DMA
>     Command Structures. The middle one of the chain is used to enable the BCH,
>     and read out the NAND page.
>
>     The WAIT4END(wait for command end) is a comunication signal between
>     the GPMI and MXS-DMA.
>
> [2] The current DMA code sets the WAIT4END bit at the last one, such as:
>
>     +-----+               +-----+                      +-----+
>     | cmd | ------------> | cmd | ------------------>  | cmd |
>     +-----+               +-----+                      +-----+
>                                                           ^
>                                                           |
>                                                           |
>                                                      set WAIT4END here
>
>     This chain works fine in the mx23/mx28.
>
> [3] But in the new GPMI version (used in MX50/MX60), the WAIT4END bit should
>     be set not only at the last DMA Command Structure,
>     but also at the middle one, such as:
>
>     +-----+               +-----+                      +-----+
>     | cmd | ------------> | cmd | ------------------>  | cmd |
>     +-----+               +-----+                      +-----+
>                              ^                            ^
>                              |                            |
>                              |                            |
>                         set WAIT4END here too        set WAIT4END here
>
>     If we do not set WAIT4END, the BCH maybe stalls in "ECC reading page" 
> state.
>     In the next ECC write page operation, a DMA-timeout occurs.
>     This has been catched in the MX6Q board.
>
> [4] In order to fix the bug, rewrite the last parameter of 
> mxs_dma_prep_slave_sg(),
>     and use the dma_ctrl_flags:
>     ---------------------------------------------------------
>       DMA_PREP_INTERRUPT : append a new DMA Command Structrue.
>       DMA_CTRL_ACK       : set the WAIT4END bit for this DMA Command 
> Structure.
>     ---------------------------------------------------------
>
> [5] changes to the relative drivers:
>     <1> For mxs-mmc driver, just use the new flags, do not change any logic.
>     <2> For gpmi-nand driver, add new field `gpmi_version` to distinguish
>         different gpmi versions, and use the new flags.
>
> Signed-off-by: Huang Shijie <[email protected]>
> ---
>  drivers/dma/mxs-dma.c                  |   32 
> ++++++++++++++++++++++++++++----
>  drivers/mmc/host/mxs-mmc.c             |   10 +++++-----
>  drivers/mtd/nand/gpmi-nand/gpmi-lib.c  |   28 ++++++++++++++++++++++------
>  drivers/mtd/nand/gpmi-nand/gpmi-nand.h |    4 ++++
>  drivers/mtd/nand/gpmi-nand/gpmi-regs.h |    2 ++
>  5 files changed, 61 insertions(+), 15 deletions(-)
>
> diff --git a/drivers/dma/mxs-dma.c b/drivers/dma/mxs-dma.c
> index c80fbed..7d7498b 100644
> --- a/drivers/dma/mxs-dma.c
> +++ b/drivers/dma/mxs-dma.c
> @@ -349,10 +349,32 @@ static void mxs_dma_free_chan_resources(struct dma_chan 
> *chan)
>       clk_disable_unprepare(mxs_dma->clk);
>  }
>  
> +/*
> + * How to use the flags for ->device_prep_slave_sg() :
> + *    [1] If there is only one DMA command in the DMA chain, the code should 
> be:
> + *            ......
> + *            ->device_prep_slave_sg(DMA_CTRL_ACK);
> + *            ......
> + *    [2] If there are two DMA commands in the DMA chain, the code should be
> + *            ......
> + *            ->device_prep_slave_sg(0);
> + *            ......
> + *            ->device_prep_slave_sg(DMA_PREP_INTERRUPT | DMA_CTRL_ACK);
> + *            ......
> + *    [3] If there are more than two DMA commands in the DMA chain, the code
> + *        should be:
> + *            ......
> + *            ->device_prep_slave_sg(0);                                // 
> First
> + *            ......
> + *            ->device_prep_slave_sg(DMA_PREP_INTERRUPT [| DMA_CTRL_ACK]);
> + *            ......
> + *            ->device_prep_slave_sg(DMA_PREP_INTERRUPT | DMA_CTRL_ACK); // 
> Last
> + *            ......
> + */
>  static struct dma_async_tx_descriptor *mxs_dma_prep_slave_sg(
>               struct dma_chan *chan, struct scatterlist *sgl,
>               unsigned int sg_len, enum dma_transfer_direction direction,
> -             unsigned long append)
> +             unsigned long flags)
>  {
>       struct mxs_dma_chan *mxs_chan = to_mxs_dma_chan(chan);
>       struct mxs_dma_engine *mxs_dma = mxs_chan->mxs_dma;
> @@ -360,6 +382,7 @@ static struct dma_async_tx_descriptor 
> *mxs_dma_prep_slave_sg(
>       struct scatterlist *sg;
>       int i, j;
>       u32 *pio;
> +     bool append = flags & DMA_PREP_INTERRUPT;
>       int idx = append ? mxs_chan->desc_count : 0;
>  
>       if (mxs_chan->status == DMA_IN_PROGRESS && !append)
> @@ -386,7 +409,6 @@ static struct dma_async_tx_descriptor 
> *mxs_dma_prep_slave_sg(
>               ccw->bits |= CCW_CHAIN;
>               ccw->bits &= ~CCW_IRQ;
>               ccw->bits &= ~CCW_DEC_SEM;
> -             ccw->bits &= ~CCW_WAIT4END;
>       } else {
>               idx = 0;
>       }
> @@ -401,7 +423,8 @@ static struct dma_async_tx_descriptor 
> *mxs_dma_prep_slave_sg(
>               ccw->bits = 0;
>               ccw->bits |= CCW_IRQ;
>               ccw->bits |= CCW_DEC_SEM;
> -             ccw->bits |= CCW_WAIT4END;
> +             if (flags & DMA_CTRL_ACK)
> +                     ccw->bits |= CCW_WAIT4END;
>               ccw->bits |= CCW_HALT_ON_TERM;
>               ccw->bits |= CCW_TERM_FLUSH;
>               ccw->bits |= BF_CCW(sg_len, PIO_NUM);
> @@ -432,7 +455,8 @@ static struct dma_async_tx_descriptor 
> *mxs_dma_prep_slave_sg(
>                               ccw->bits &= ~CCW_CHAIN;
>                               ccw->bits |= CCW_IRQ;
>                               ccw->bits |= CCW_DEC_SEM;
> -                             ccw->bits |= CCW_WAIT4END;
> +                             if (flags & DMA_CTRL_ACK)
> +                                     ccw->bits |= CCW_WAIT4END;
>                       }
>               }
>       }
> diff --git a/drivers/mmc/host/mxs-mmc.c b/drivers/mmc/host/mxs-mmc.c
> index 1e18970..120669b 100644
> --- a/drivers/mmc/host/mxs-mmc.c
> +++ b/drivers/mmc/host/mxs-mmc.c
> @@ -305,7 +305,7 @@ static irqreturn_t mxs_mmc_irq_handler(int irq, void 
> *dev_id)
>  }
>  
>  static struct dma_async_tx_descriptor *mxs_mmc_prep_dma(
> -     struct mxs_mmc_host *host, unsigned int append)
> +     struct mxs_mmc_host *host, unsigned long flags)
>  {
>       struct dma_async_tx_descriptor *desc;
>       struct mmc_data *data = host->data;
> @@ -325,7 +325,7 @@ static struct dma_async_tx_descriptor *mxs_mmc_prep_dma(
>       }
>  
>       desc = host->dmach->device->device_prep_slave_sg(host->dmach,
> -                             sgl, sg_len, host->slave_dirn, append);
> +                             sgl, sg_len, host->slave_dirn, flags);
>       if (desc) {
>               desc->callback = mxs_mmc_dma_irq_callback;
>               desc->callback_param = host;
> @@ -358,7 +358,7 @@ static void mxs_mmc_bc(struct mxs_mmc_host *host)
>       host->ssp_pio_words[2] = cmd1;
>       host->dma_dir = DMA_NONE;
>       host->slave_dirn = DMA_TRANS_NONE;
> -     desc = mxs_mmc_prep_dma(host, 0);
> +     desc = mxs_mmc_prep_dma(host, DMA_CTRL_ACK);
>       if (!desc)
>               goto out;
>  
> @@ -398,7 +398,7 @@ static void mxs_mmc_ac(struct mxs_mmc_host *host)
>       host->ssp_pio_words[2] = cmd1;
>       host->dma_dir = DMA_NONE;
>       host->slave_dirn = DMA_TRANS_NONE;
> -     desc = mxs_mmc_prep_dma(host, 0);
> +     desc = mxs_mmc_prep_dma(host, DMA_CTRL_ACK);
>       if (!desc)
>               goto out;
>  
> @@ -526,7 +526,7 @@ static void mxs_mmc_adtc(struct mxs_mmc_host *host)
>       host->data = data;
>       host->dma_dir = dma_data_dir;
>       host->slave_dirn = slave_dirn;
> -     desc = mxs_mmc_prep_dma(host, 1);
> +     desc = mxs_mmc_prep_dma(host, DMA_PREP_INTERRUPT | DMA_CTRL_ACK);
>       if (!desc)
>               goto out;
>  
> diff --git a/drivers/mtd/nand/gpmi-nand/gpmi-lib.c 
> b/drivers/mtd/nand/gpmi-nand/gpmi-lib.c
> index 7f68042..2029995 100644
> --- a/drivers/mtd/nand/gpmi-nand/gpmi-lib.c
> +++ b/drivers/mtd/nand/gpmi-nand/gpmi-lib.c
> @@ -133,6 +133,9 @@ int gpmi_init(struct gpmi_nand_data *this)
>       if (ret)
>               goto err_out;
>  
> +     /* Read out the GPMI version */
> +     this->gpmi_version = readl(r->gpmi_regs + HW_GPMI_VERSION);
> +
>       /* Choose NAND mode. */
>       writel(BM_GPMI_CTRL1_GPMI_MODE, r->gpmi_regs + HW_GPMI_CTRL1_CLR);
>  
> @@ -839,7 +842,9 @@ int gpmi_send_command(struct gpmi_nand_data *this)
>       sg_init_one(sgl, this->cmd_buffer, this->command_length);
>       dma_map_sg(this->dev, sgl, 1, DMA_TO_DEVICE);
>       desc = channel->device->device_prep_slave_sg(channel,
> -                                     sgl, 1, DMA_MEM_TO_DEV, 1);
> +                             sgl, 1, DMA_MEM_TO_DEV,
> +                             DMA_PREP_INTERRUPT | DMA_CTRL_ACK);
> +
>       if (!desc) {
>               pr_err("step 2 error\n");
>               return -1;
> @@ -881,7 +886,8 @@ int gpmi_send_data(struct gpmi_nand_data *this)
>       /* [2] send DMA request */
>       prepare_data_dma(this, DMA_TO_DEVICE);
>       desc = channel->device->device_prep_slave_sg(channel, &this->data_sgl,
> -                                             1, DMA_MEM_TO_DEV, 1);
> +                                     1, DMA_MEM_TO_DEV,
> +                                     DMA_PREP_INTERRUPT | DMA_CTRL_ACK);
>       if (!desc) {
>               pr_err("step 2 error\n");
>               return -1;
> @@ -917,7 +923,8 @@ int gpmi_read_data(struct gpmi_nand_data *this)
>       /* [2] : send DMA request */
>       prepare_data_dma(this, DMA_FROM_DEVICE);
>       desc = channel->device->device_prep_slave_sg(channel, &this->data_sgl,
> -                                             1, DMA_DEV_TO_MEM, 1);
> +                                     1, DMA_DEV_TO_MEM,
> +                                     DMA_PREP_INTERRUPT | DMA_CTRL_ACK);
>       if (!desc) {
>               pr_err("step 2 error\n");
>               return -1;
> @@ -964,7 +971,8 @@ int gpmi_send_page(struct gpmi_nand_data *this,
>  
>       desc = channel->device->device_prep_slave_sg(channel,
>                                       (struct scatterlist *)pio,
> -                                     ARRAY_SIZE(pio), DMA_TRANS_NONE, 0);
> +                                     ARRAY_SIZE(pio), DMA_TRANS_NONE,
> +                                     DMA_CTRL_ACK);
>       if (!desc) {
>               pr_err("step 2 error\n");
>               return -1;
> @@ -984,6 +992,7 @@ int gpmi_read_page(struct gpmi_nand_data *this,
>       struct dma_async_tx_descriptor *desc;
>       struct dma_chan *channel = get_dma_chan(this);
>       int chip = this->current_chip;
> +     unsigned long flags;
>       u32 pio[6];
>  
>       /* [1] Wait for the chip to report ready. */
> @@ -1026,9 +1035,14 @@ int gpmi_read_page(struct gpmi_nand_data *this,
>       pio[3] = geo->page_size;
>       pio[4] = payload;
>       pio[5] = auxiliary;
> +
> +     /* Check GPMI's version, set DMA_CTRL_ACK if needed. */
> +     flags = DMA_PREP_INTERRUPT;
> +     if (this->gpmi_version == GPMI_VERSION_0501)
> +             flags |= DMA_CTRL_ACK;
>       desc = channel->device->device_prep_slave_sg(channel,
>                                       (struct scatterlist *)pio,
> -                                     ARRAY_SIZE(pio), DMA_TRANS_NONE, 1);
> +                                     ARRAY_SIZE(pio), DMA_TRANS_NONE, flags);
>       if (!desc) {
>               pr_err("step 2 error\n");
>               return -1;
> @@ -1045,9 +1059,11 @@ int gpmi_read_page(struct gpmi_nand_data *this,
>               | BF_GPMI_CTRL0_ADDRESS(address)
>               | BF_GPMI_CTRL0_XFER_COUNT(geo->page_size);
>       pio[1] = 0;
> +     pio[2] = 0;
>       desc = channel->device->device_prep_slave_sg(channel,
>                               (struct scatterlist *)pio, 2,
> -                             DMA_TRANS_NONE, 1);
> +                             DMA_TRANS_NONE,
> +                             DMA_PREP_INTERRUPT | DMA_CTRL_ACK);
>       if (!desc) {
>               pr_err("step 3 error\n");
>               return -1;
> diff --git a/drivers/mtd/nand/gpmi-nand/gpmi-nand.h 
> b/drivers/mtd/nand/gpmi-nand/gpmi-nand.h
> index 1c7fdbb..5c277e3 100644
> --- a/drivers/mtd/nand/gpmi-nand/gpmi-nand.h
> +++ b/drivers/mtd/nand/gpmi-nand/gpmi-nand.h
> @@ -132,6 +132,10 @@ struct gpmi_nand_data {
>       /* Flash Hardware */
>       struct nand_timing      timing;
>  
> +     /* GPMI hardware version */
> +#define GPMI_VERSION_0501    (0x05010000)
> +     u32                     gpmi_version;
> +
>       /* BCH */
>       struct bch_geometry     bch_geometry;
>       struct completion       bch_done;
> diff --git a/drivers/mtd/nand/gpmi-nand/gpmi-regs.h 
> b/drivers/mtd/nand/gpmi-nand/gpmi-regs.h
> index 8343124..f005b24 100644
> --- a/drivers/mtd/nand/gpmi-nand/gpmi-regs.h
> +++ b/drivers/mtd/nand/gpmi-nand/gpmi-regs.h
> @@ -169,4 +169,6 @@
>  #define HW_GPMI_DEBUG                                        0x000000c0
>  #define MX23_BP_GPMI_DEBUG_READY0                    28
>  #define MX23_BM_GPMI_DEBUG_READY0    (1 << MX23_BP_GPMI_DEBUG_READY0)
> +
> +#define HW_GPMI_VERSION                                      0x000000d0
>  #endif


--
To unsubscribe from this list: send the line "unsubscribe linux-mmc" in
the body of a message to [email protected]
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to