[PATCH 2/2] mtd: atmel_nand: NFC: support multiple interrupt handling

Bo Shen voice.shen at atmel.com
Thu Jun 5 23:17:08 PDT 2014


Hi Josh,

On 06/06/2014 11:48 AM, Josh Wu wrote:
> It fixes following error sometime happens during the NFC data transfer:
>    atmel_nand 80000000.nand: Time out to wait for interrupt: 0x00010000
>    atmel_nand 80000000.nand: something wrong, No XFR_DONE interrupt comes.
>
> The root cause is in current interrupt handler, we read the ISR but
> only handle one interrupt. If there are more than one interrupt come
> together, then the second one will be lost.
>
> During the NFC data transfer. There is possible two NFC interrupts:
> NFC_CMD_DONE and NFC_XFR_DONE, come in same time.
>
> NFC_CMD_DONE means NFC command is send. and NFC_XFR_DONE means NFC data
> is transfered.
>
> This patch can handle multiple NFC interrupts in same time. And during
> the NFC data transfer, we need to wait for two NFC interrupts:
> NFC_CMD_DONE and NFC_XFR_DONE.

I think, these two patches is try to fix this issue, can you combine it 
into one? Or else, I really don't know what the patch 1 is benefit for?

> Reported-by: Matthieu CRAPET <Matthieu.CRAPET at ingenico.com>
> Signed-off-by: Josh Wu <josh.wu at atmel.com>
> ---
>   drivers/mtd/nand/atmel_nand.c |   61 +++++++++++++++++++++++------------------
>   1 file changed, 35 insertions(+), 26 deletions(-)
>
> diff --git a/drivers/mtd/nand/atmel_nand.c b/drivers/mtd/nand/atmel_nand.c
> index 347cee2..faee53d 100644
> --- a/drivers/mtd/nand/atmel_nand.c
> +++ b/drivers/mtd/nand/atmel_nand.c
> @@ -1579,7 +1579,7 @@ static irqreturn_t hsmc_interrupt(int irq, void *dev_id)
>   {
>   	struct atmel_nand_host *host = dev_id;
>   	u32 status, mask, pending;
> -	irqreturn_t ret = IRQ_HANDLED;
> +	irqreturn_t ret = IRQ_NONE;
>
>   	status = nfc_readl(host->nfc->hsmc_regs, SR);
>   	mask = nfc_readl(host->nfc->hsmc_regs, IMR);
> @@ -1588,14 +1588,17 @@ static irqreturn_t hsmc_interrupt(int irq, void *dev_id)
>   	if (pending & NFC_SR_XFR_DONE) {
>   		complete(&host->nfc->comp_xfer_done);
>   		nfc_writel(host->nfc->hsmc_regs, IDR, NFC_SR_XFR_DONE);
> -	} else if (pending & NFC_SR_RB_EDGE) {
> +		ret = IRQ_HANDLED;
> +	}
> +	if (pending & NFC_SR_RB_EDGE) {
>   		complete(&host->nfc->comp_ready);
>   		nfc_writel(host->nfc->hsmc_regs, IDR, NFC_SR_RB_EDGE);
> -	} else if (pending & NFC_SR_CMD_DONE) {
> +		ret = IRQ_HANDLED;
> +	}
> +	if (pending & NFC_SR_CMD_DONE) {
>   		complete(&host->nfc->comp_cmd_done);
>   		nfc_writel(host->nfc->hsmc_regs, IDR, NFC_SR_CMD_DONE);
> -	} else {
> -		ret = IRQ_NONE;
> +		ret = IRQ_HANDLED;
>   	}
>
>   	return ret;
> @@ -1604,31 +1607,40 @@ static irqreturn_t hsmc_interrupt(int irq, void *dev_id)
>   /* NFC(Nand Flash Controller) related functions */
>   static int nfc_wait_interrupt(struct atmel_nand_host *host, u32 flag)
>   {
> -	unsigned long timeout;
> -	struct completion *comp;
> -
> -	if (flag & NFC_SR_XFR_DONE) {
> -		comp = &host->nfc->comp_xfer_done;
> -	} else if (flag & NFC_SR_RB_EDGE) {
> -		comp = &host->nfc->comp_ready;
> -	} else if (flag & NFC_SR_CMD_DONE) {
> -		comp = &host->nfc->comp_cmd_done;
> -	} else {
> +	int i, index = 0;
> +	struct completion *comp[3];	/* Support 3 interrupt completion */
> +
> +	if (flag & NFC_SR_XFR_DONE)
> +		comp[index++] = &host->nfc->comp_xfer_done;
> +
> +	if (flag & NFC_SR_RB_EDGE)
> +		comp[index++] = &host->nfc->comp_ready;
> +
> +	if (flag & NFC_SR_CMD_DONE)
> +		comp[index++] = &host->nfc->comp_cmd_done;
> +
> +	if (index == 0) {
>   		dev_err(host->dev, "Unkown interrupt flag: 0x%08x\n", flag);
>   		return -EINVAL;
>   	}
>
> -	init_completion(comp);
> +	for (i = 0; i < index; i++)
> +		init_completion(comp[i]);

One question, will the interrupt occur before you initial the 
completion? If so, you will lose the interrupt.

>
>   	/* Enable interrupt that need to wait for */
>   	nfc_writel(host->nfc->hsmc_regs, IER, flag);
>
> -	timeout = wait_for_completion_timeout(comp,
> -			msecs_to_jiffies(NFC_TIME_OUT_MS));
> -	if (timeout)
> -		return 0;
> +	for (i = 0; i < index; i++) {
> +		if (wait_for_completion_timeout(comp[i],
> +				msecs_to_jiffies(NFC_TIME_OUT_MS)))
> +			continue;	/* wait for next completion */
> +		else
> +			goto err_timeout;
> +	}
>
> -	/* Time out to wait for the interrupt */
> +	return 0;
> +
> +err_timeout:
>   	dev_err(host->dev, "Time out to wait for interrupt: 0x%08x\n", flag);
>   	return -ETIMEDOUT;
>   }
> @@ -1652,7 +1664,8 @@ static int nfc_send_command(struct atmel_nand_host *host,
>   	}
>   	nfc_writel(host->nfc->hsmc_regs, CYCLE0, cycle0);
>   	nfc_cmd_addr1234_writel(cmd, addr, host->nfc->base_cmd_regs);
> -	return nfc_wait_interrupt(host, NFC_SR_CMD_DONE);
> +	return nfc_wait_interrupt(host, NFC_SR_CMD_DONE |
> +			(cmd & NFCADDR_CMD_DATAEN ? NFC_SR_XFR_DONE : 0));
>   }
>
>   static int nfc_device_ready(struct mtd_info *mtd)
> @@ -1810,10 +1823,6 @@ static void nfc_nand_command(struct mtd_info *mtd, unsigned int command,
>   	nfc_addr_cmd = cmd1 | cmd2 | vcmd2 | acycle | csid | dataen | nfcwr;
>   	nfc_send_command(host, nfc_addr_cmd, addr1234, cycle0);
>
> -	if (dataen == NFCADDR_CMD_DATAEN)
> -		if (nfc_wait_interrupt(host, NFC_SR_XFR_DONE))
> -			dev_err(host->dev, "something wrong, No XFR_DONE interrupt comes.\n");
> -
>   	/*
>   	 * Program and erase have their own busy handlers status, sequential
>   	 * in, and deplete1 need no delay.
>

Best Regards,
Bo Shen



More information about the linux-arm-kernel mailing list