lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [day] [month] [year] [list]
Message-Id: <200805301457.34320.marc.pignat@hevs.ch>
Date:	Fri, 30 May 2008 14:57:33 +0200
From:	Marc Pignat <marc.pignat@...s.ch>
To:	Nicolas Ferre <nicolas.ferre@...el.com>
Cc:	Pierre Ossman <drzeus@...eus.cx>,
	"Hans-Jürgen "@rfoatmel.rfo.atmel.com,
	Linux Kernel list <linux-kernel@...r.kernel.org>,
	Andrew Victor <linux@...im.org.za>,
	Eric BENARD <ebenard@...e.fr>,
	"Ville Syrjä"@rfoatmel.rfo.atmel.com
Subject: Re: [PATCH 1/7] mmc: at91_mci: support for block size not modulo 4

Hi Nicolas!

On Friday 30 May 2008, Nicolas Ferre wrote:
> From: Marc Pignat <marc.pignat@...s.ch>
> 
> Implement transfer with size not modulo 4 for at91sam9*. Please note that the
> at91rm9200 simply can't handle this.
> 
> Signed-off-by: Marc Pignat <marc.pignat@...s.ch>
> Signed-off-by: Nicolas Ferre <nicolas.ferre@...el.com>
> ---

I just want we don't forget that at91sam9260 errata, pointed by someone when I
posted this patch...

44.2.4.4 Data Write Operation and number of bytes
                     The Data Write operation with a number of bytes less than 12 is impossible.
                     Problem Fix/Workaround
                     The PDC counters must always be equal to 12 bytes for data transfers lower than 12 bytes. The
                     BLKLEN or BCNT field are used to specify the real count number.

We should probably fix that thing in the next release and verify if other
chips are affected!



>  drivers/mmc/host/at91_mci.c |   15 ++++++++++-----
>  1 files changed, 10 insertions(+), 5 deletions(-)
> 
> diff --git a/drivers/mmc/host/at91_mci.c b/drivers/mmc/host/at91_mci.c
> index 8979ad3..543b64b 100644
> --- a/drivers/mmc/host/at91_mci.c
> +++ b/drivers/mmc/host/at91_mci.c
> @@ -233,11 +233,11 @@ static void at91_mci_pre_dma_read(struct at91mci_host *host)
>  
>               if (i == 0) {
>                       at91_mci_write(host, ATMEL_PDC_RPR, sg->dma_address);
> -                     at91_mci_write(host, ATMEL_PDC_RCR, sg->length / 4);
> +                     at91_mci_write(host, ATMEL_PDC_RCR, (data->blksz & 0x3) ? sg->length : sg->length / 4);
>               }
>               else {
>                       at91_mci_write(host, ATMEL_PDC_RNPR, sg->dma_address);
> -                     at91_mci_write(host, ATMEL_PDC_RNCR, sg->length / 4);
> +                     at91_mci_write(host, ATMEL_PDC_RNCR, (data->blksz & 0x3) ? sg->length : sg->length / 4);
>               }
>       }
>  
> @@ -430,7 +430,7 @@ static void at91_mci_send_command(struct at91mci_host *host, struct mmc_command
>  
>       if (data) {
>  
> -             if ( data->blksz & 0x3 ) {
> +             if ( cpu_is_at91rm9200() && (data->blksz & 0x3) ) {
>                       pr_debug("Unsupported block size\n");
>                       cmd->error = -EINVAL;
>                       mmc_request_done(host->mmc, host->request);
> @@ -482,7 +482,10 @@ static void at91_mci_send_command(struct at91mci_host *host, struct mmc_command
>       } else {
>               /* zero block length and PDC mode */
>               mr = at91_mci_read(host, AT91_MCI_MR) & 0x7fff;
> -             at91_mci_write(host, AT91_MCI_MR, mr | (block_length << 16) | AT91_MCI_PDCMODE);
here, we should do something like
{
        /* blocks smaller than 16 will be rounded up to 16 at91sam9260, and 16 is modulo 4 :) */
        if (!(cpu_is_at91sam9260() && size < 16))
                mr |= (data->blksz & 0x3) ? AT91_MCI_PDCFBYTE : 0;
}
> +             mr |= (block_length << 16);
> +             mr |= AT91_MCI_PDCMODE;
> +             at91_mci_write(host, AT91_MCI_MR, mr);
>  
>               /*
>                * Disable the PDC controller
> @@ -517,7 +520,9 @@ static void at91_mci_send_command(struct at91mci_host *host, struct mmc_command
>                               pr_debug("Transmitting %d bytes\n", host->total_length);
>  
>                               at91_mci_write(host, ATMEL_PDC_TPR, host->physical_address);
> -                             at91_mci_write(host, ATMEL_PDC_TCR, host->total_length / 4);
> +                             at91_mci_write(host, ATMEL_PDC_TCR, (data->blksz & 0x3) ?
> +                                             host->total_length : host->total_length / 4);
here, we should add something like
{
        unsigned long size = data->blksz;
        if (cpu_is_at91sam9260() && size < 16 )
                size = 16;
        at91_mci_write(host, ATMEL_PDC_TCR, (size & 0x3) ? size : size / 4);
}
> +
>                               ier = AT91_MCI_CMDRDY;
>                       }
>               }

Best regards

Marc

Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ