linux-ide.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [PATCH] scc_pata.c: Workaround for errata A308
@ 2007-07-10  9:29 Akira Iguchi
  2007-07-13 16:50 ` Sergei Shtylyov
  0 siblings, 1 reply; 3+ messages in thread
From: Akira Iguchi @ 2007-07-10  9:29 UTC (permalink / raw)
  To: bzolnier; +Cc: kou.ishizaki, linux-ide

Workaround for errata A308: turn down the UDMA mode and retry
the DMA command when the data lost condition is detected.

Signed-off-by: Kou Ishizaki <kou.ishizaki@toshiba.co.jp>
Signed-off-by: Akira Iguchi <akira2.iguchi@toshiba.co.jp>
---

diff -purN -X linux-powerpc-git/Documentation/dontdiff linux-powerpc-git/drivers/ide/pci/scc_pata.c linux-powerpc-git.mod/drivers/ide/pci/scc_pata.c
--- linux-powerpc-git/drivers/ide/pci/scc_pata.c	2007-05-22 13:08:32.000000000 +0900
+++ linux-powerpc-git.mod/drivers/ide/pci/scc_pata.c	2007-07-10 17:10:10.000000000 +0900
@@ -278,6 +278,13 @@ static int scc_tune_chipset(ide_drive_t 
 		offset = 0; /* 100MHz */
 	}
 
+	/* errata A308 workaround: limit non ide_disk drive to UDMA4 */
+	if (drive->media != ide_disk && speed > XFER_UDMA_4) {
+		printk(KERN_INFO "%s: limit drive(media=%x) to UDMA4\n",
+		       SCC_PATA_NAME, drive->media);
+		speed = XFER_UDMA_4;
+	}
+
 	switch (speed) {
 	case XFER_UDMA_6:
 		idx = 6;
@@ -401,6 +408,33 @@ static int scc_ide_dma_end(ide_drive_t *
 	ide_hwif_t *hwif = HWIF(drive);
 	unsigned long intsts_port = hwif->dma_base + 0x014;
 	u32 reg;
+	int dma_stat, data_lost = 0;
+	static int retry = 0;
+
+	/* errata A308 workaround: Step5 (check data lost) */
+	/* We don't check non ide_disk because it is limited to UDMA4 */
+	if (!(in_be32((void __iomem *)IDE_ALTSTATUS_REG) & ERR_STAT) &&
+	    drive->media == ide_disk && drive->current_speed > XFER_UDMA_4) {
+		reg = in_be32((void __iomem *)intsts_port);
+		if (!(reg & INTSTS_ACTEINT)) {
+			printk(KERN_WARNING "%s: data lost occurred. (ACTEINT==0, retry:%d)\n",
+			       drive->name, retry);
+			data_lost = 1;
+			if (retry++) {
+				struct request *rq = HWGROUP(drive)->rq;
+				int unit;
+				/* ERROR_RESET and drive->crc_count are needed
+				 * to reduce DMA transfer mode in retry process.
+				 */
+				if (rq)
+					rq->errors |= ERROR_RESET;
+				for (unit = 0; unit < MAX_DRIVES; unit++) {
+					ide_drive_t *drive = &hwif->drives[unit];
+					drive->crc_count++;
+				}
+			}
+		}
+	}
 
 	while (1) {
 		reg = in_be32((void __iomem *)intsts_port);
@@ -469,27 +503,25 @@ static int scc_ide_dma_end(ide_drive_t *
 		break;
 	}
 
-	return __ide_dma_end(drive);
+	dma_stat = __ide_dma_end(drive);
+	if (data_lost)
+		dma_stat |= 2; /* emulate DMA error (to retry command) */
+	return dma_stat;
 }
 
 /* returns 1 if dma irq issued, 0 otherwise */
 static int scc_dma_test_irq(ide_drive_t *drive)
 {
-	ide_hwif_t *hwif	= HWIF(drive);
-	u8 dma_stat		= hwif->INB(hwif->dma_status);
+	ide_hwif_t *hwif = HWIF(drive);
+	u32 int_stat = in_be32((void __iomem *)hwif->dma_base + 0x014);
 
-	/* return 1 if INTR asserted */
-	if ((dma_stat & 4) == 4)
+	/* SCC errata A252,A308 workaround: Step4 */
+	if (in_be32((void __iomem *)IDE_ALTSTATUS_REG) & ERR_STAT &&
+	    int_stat & INTSTS_INTRQ)
 		return 1;
 
-	/* Workaround for PTERADD: emulate DMA_INTR when
-	 * - IDE_STATUS[ERR] = 1
-	 * - INT_STATUS[INTRQ] = 1
-	 * - DMA_STATUS[IORACTA] = 1
-	 */
-	if (in_be32((void __iomem *)IDE_ALTSTATUS_REG) & ERR_STAT &&
-	    in_be32((void __iomem *)(hwif->dma_base + 0x014)) & INTSTS_INTRQ &&
-		dma_stat & 1)
+	/* SCC errata A308 workaround: Step5 (polling IOIRQS) */
+	if (int_stat & INTSTS_IOIRQS)
 		return 1;
 
 	if (!drive->waiting_for_dma)


^ permalink raw reply	[flat|nested] 3+ messages in thread

* Re: [PATCH] scc_pata.c: Workaround for errata A308
       [not found] <200707100929.l6A9T95v017616@toshiba.co.jp>
@ 2007-07-10 21:58 ` Bartlomiej Zolnierkiewicz
  0 siblings, 0 replies; 3+ messages in thread
From: Bartlomiej Zolnierkiewicz @ 2007-07-10 21:58 UTC (permalink / raw)
  To: Akira Iguchi; +Cc: kou.ishizaki, linux-ide

On Tuesday 10 July 2007, Akira Iguchi wrote:
> Workaround for errata A308: turn down the UDMA mode and retry
> the DMA command when the data lost condition is detected.
> 
> Signed-off-by: Kou Ishizaki <kou.ishizaki@toshiba.co.jp>
> Signed-off-by: Akira Iguchi <akira2.iguchi@toshiba.co.jp>

applied, thanks

^ permalink raw reply	[flat|nested] 3+ messages in thread

* Re: [PATCH] scc_pata.c: Workaround for errata A308
  2007-07-10  9:29 Akira Iguchi
@ 2007-07-13 16:50 ` Sergei Shtylyov
  0 siblings, 0 replies; 3+ messages in thread
From: Sergei Shtylyov @ 2007-07-13 16:50 UTC (permalink / raw)
  To: Akira Iguchi; +Cc: bzolnier, kou.ishizaki, linux-ide

Hello.

Akira Iguchi wrote:
> Workaround for errata A308: turn down the UDMA mode and retry
> the DMA command when the data lost condition is detected.

> Signed-off-by: Kou Ishizaki <kou.ishizaki@toshiba.co.jp>
> Signed-off-by: Akira Iguchi <akira2.iguchi@toshiba.co.jp>
> ---

> diff -purN -X linux-powerpc-git/Documentation/dontdiff linux-powerpc-git/drivers/ide/pci/scc_pata.c linux-powerpc-git.mod/drivers/ide/pci/scc_pata.c
> --- linux-powerpc-git/drivers/ide/pci/scc_pata.c	2007-05-22 13:08:32.000000000 +0900
> +++ linux-powerpc-git.mod/drivers/ide/pci/scc_pata.c	2007-07-10 17:10:10.000000000 +0900
> @@ -278,6 +278,13 @@ static int scc_tune_chipset(ide_drive_t 
>  		offset = 0; /* 100MHz */
>  	}
>  
> +	/* errata A308 workaround: limit non ide_disk drive to UDMA4 */
> +	if (drive->media != ide_disk && speed > XFER_UDMA_4) {
> +		printk(KERN_INFO "%s: limit drive(media=%x) to UDMA4\n",
> +		       SCC_PATA_NAME, drive->media);
> +		speed = XFER_UDMA_4;
> +	}
> +

    The same comment as Alan had to the libata driver: use the new 
udma_filter() method for this.

> @@ -401,6 +408,33 @@ static int scc_ide_dma_end(ide_drive_t *
>  	ide_hwif_t *hwif = HWIF(drive);
>  	unsigned long intsts_port = hwif->dma_base + 0x014;
>  	u32 reg;
> +	int dma_stat, data_lost = 0;
> +	static int retry = 0;
> +
> +	/* errata A308 workaround: Step5 (check data lost) */
> +	/* We don't check non ide_disk because it is limited to UDMA4 */
> +	if (!(in_be32((void __iomem *)IDE_ALTSTATUS_REG) & ERR_STAT) &&
> +	    drive->media == ide_disk && drive->current_speed > XFER_UDMA_4) {
> +		reg = in_be32((void __iomem *)intsts_port);
> +		if (!(reg & INTSTS_ACTEINT)) {
> +			printk(KERN_WARNING "%s: data lost occurred. (ACTEINT==0, retry:%d)\n",

    Should be "data loss" -- if this indeed means that data was lost *on the 
media*.

> +			       drive->name, retry);
> +			data_lost = 1;
> +			if (retry++) {
> +				struct request *rq = HWGROUP(drive)->rq;
> +				int unit;
> +				/* ERROR_RESET and drive->crc_count are needed
> +				 * to reduce DMA transfer mode in retry process.
> +				 */
> +				if (rq)
> +					rq->errors |= ERROR_RESET;
> +				for (unit = 0; unit < MAX_DRIVES; unit++) {
> +					ide_drive_t *drive = &hwif->drives[unit];
> +					drive->crc_count++;
> +				}
> +			}
> +		}
> +	}
>  
>  	while (1) {
>  		reg = in_be32((void __iomem *)intsts_port);
> @@ -469,27 +503,25 @@ static int scc_ide_dma_end(ide_drive_t *
>  		break;
>  	}
>  
> -	return __ide_dma_end(drive);
> +	dma_stat = __ide_dma_end(drive);
> +	if (data_lost)
> +		dma_stat |= 2; /* emulate DMA error (to retry command) */
> +	return dma_stat;
>  }
>  
>  /* returns 1 if dma irq issued, 0 otherwise */
>  static int scc_dma_test_irq(ide_drive_t *drive)
>  {
> -	ide_hwif_t *hwif	= HWIF(drive);
> -	u8 dma_stat		= hwif->INB(hwif->dma_status);
> +	ide_hwif_t *hwif = HWIF(drive);
> +	u32 int_stat = in_be32((void __iomem *)hwif->dma_base + 0x014);
>  
> -	/* return 1 if INTR asserted */
> -	if ((dma_stat & 4) == 4)
> +	/* SCC errata A252,A308 workaround: Step4 */
> +	if (in_be32((void __iomem *)IDE_ALTSTATUS_REG) & ERR_STAT &&

    Enclose bitwise AND expression into parens, please.

> +	    int_stat & INTSTS_INTRQ)
>  		return 1;
>  
> -	/* Workaround for PTERADD: emulate DMA_INTR when
> -	 * - IDE_STATUS[ERR] = 1
> -	 * - INT_STATUS[INTRQ] = 1
> -	 * - DMA_STATUS[IORACTA] = 1
> -	 */
> -	if (in_be32((void __iomem *)IDE_ALTSTATUS_REG) & ERR_STAT &&
> -	    in_be32((void __iomem *)(hwif->dma_base + 0x014)) & INTSTS_INTRQ &&
> -		dma_stat & 1)
> +	/* SCC errata A308 workaround: Step5 (polling IOIRQS) */
> +	if (int_stat & INTSTS_IOIRQS)
>  		return 1;
>  
>  	if (!drive->waiting_for_dma)

MBR, Sergei

^ permalink raw reply	[flat|nested] 3+ messages in thread

end of thread, other threads:[~2007-07-13 16:48 UTC | newest]

Thread overview: 3+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
     [not found] <200707100929.l6A9T95v017616@toshiba.co.jp>
2007-07-10 21:58 ` [PATCH] scc_pata.c: Workaround for errata A308 Bartlomiej Zolnierkiewicz
2007-07-10  9:29 Akira Iguchi
2007-07-13 16:50 ` Sergei Shtylyov

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).