Hi,

Thank you for that patch, a few questions/comments below.

On 28/07/2014 at 13:43:40 +0200, Jiri Prchal wrote :
> This fix problem with PA14 set as periph A left from ROMBOOT and need it
> to be set to gpio before spi bus communicate with chip on CS0 on other
> gpio. As I reported a week ago.
> Request all csgpios in spi_probe since cs depends on each other and must
> be all of them in right state before any communication on bus. They are
> part of bus, like miso, mosi, clk, not part of chips, they are defined
> in parent spi node, not in child chip node.
> Csgpio moved to atmel_spi struct as part of master bus.
> 
> Signed-off-by: Jiri Prchal <[email protected]>
> ---
>  drivers/spi/spi-atmel.c | 68 
> +++++++++++++++++++++++++++++--------------------
>  1 file changed, 41 insertions(+), 27 deletions(-)
> 
> diff --git a/drivers/spi/spi-atmel.c b/drivers/spi/spi-atmel.c
> index 92a6f0d..1fb7bb9 100644
> --- a/drivers/spi/spi-atmel.c
> +++ b/drivers/spi/spi-atmel.c
> @@ -22,11 +22,14 @@
>  #include <linux/platform_data/atmel.h>
>  #include <linux/platform_data/dma-atmel.h>
>  #include <linux/of.h>
> +#include <linux/of_gpio.h>
>  
>  #include <linux/io.h>
>  #include <linux/gpio.h>
>  #include <linux/pinctrl/consumer.h>
>  
> +#define DRIVER_NAME "atmel-spi"
> +

This is not really related to the issue solved by that patch, maybe it
could go in another patch ?

>  /* SPI register offsets */
>  #define SPI_CR                                       0x0000
>  #define SPI_MR                                       0x0004
> @@ -242,11 +245,13 @@ struct atmel_spi {
>  
>       bool                    keep_cs;
>       bool                    cs_active;
> +
> +     /* chip selects */
> +     unsigned int            csgpio[];
>  };
>  
>  /* Controller-specific per-slave state */
>  struct atmel_spi_device {
> -     unsigned int            npcs_pin;
>       u32                     csr;
>  };
>  
> @@ -312,7 +317,7 @@ static void cs_activate(struct atmel_spi *as, struct 
> spi_device *spi)
>               }
>  
>               mr = spi_readl(as, MR);
> -             gpio_set_value(asd->npcs_pin, active);
> +             gpio_set_value(as->csgpio[spi->chip_select], active);
>       } else {
>               u32 cpol = (spi->mode & SPI_CPOL) ? SPI_BIT(CPOL) : 0;
>               int i;
> @@ -329,18 +334,16 @@ static void cs_activate(struct atmel_spi *as, struct 
> spi_device *spi)
>               mr = spi_readl(as, MR);
>               mr = SPI_BFINS(PCS, ~(1 << spi->chip_select), mr);
>               if (spi->chip_select != 0)
> -                     gpio_set_value(asd->npcs_pin, active);
> +                     gpio_set_value(as->csgpio[spi->chip_select], active);
>               spi_writel(as, MR, mr);
>       }
>  
>       dev_dbg(&spi->dev, "activate %u%s, mr %08x\n",
> -                     asd->npcs_pin, active ? " (high)" : "",
> -                     mr);
> +             as->csgpio[spi->chip_select], active ? " (high)" : "", mr);
>  }
>  
>  static void cs_deactivate(struct atmel_spi *as, struct spi_device *spi)
>  {
> -     struct atmel_spi_device *asd = spi->controller_state;
>       unsigned active = spi->mode & SPI_CS_HIGH;
>       u32 mr;
>  
> @@ -354,11 +357,10 @@ static void cs_deactivate(struct atmel_spi *as, struct 
> spi_device *spi)
>       }
>  
>       dev_dbg(&spi->dev, "DEactivate %u%s, mr %08x\n",
> -                     asd->npcs_pin, active ? " (low)" : "",
> -                     mr);
> +             as->csgpio[spi->chip_select], active ? " (low)" : "", mr);
>  
>       if (atmel_spi_is_v2(as) || spi->chip_select != 0)
> -             gpio_set_value(asd->npcs_pin, !active);
> +             gpio_set_value(as->csgpio[spi->chip_select], !active);
>  }
>  
>  static void atmel_spi_lock(struct atmel_spi *as) __acquires(&as->lock)
> @@ -989,8 +991,6 @@ static int atmel_spi_setup(struct spi_device *spi)
>       struct atmel_spi_device *asd;
>       u32                     csr;
>       unsigned int            bits = spi->bits_per_word;
> -     unsigned int            npcs_pin;
> -     int                     ret;
>  
>       as = spi_master_get_devdata(spi->master);
>  
> @@ -1017,27 +1017,13 @@ static int atmel_spi_setup(struct spi_device *spi)
>       csr |= SPI_BF(DLYBS, 0);
>       csr |= SPI_BF(DLYBCT, 0);
>  
> -     /* chipselect must have been muxed as GPIO (e.g. in board setup) */
> -     npcs_pin = (unsigned int)spi->controller_data;
> -
> -     if (gpio_is_valid(spi->cs_gpio))
> -             npcs_pin = spi->cs_gpio;
> -
>       asd = spi->controller_state;
>       if (!asd) {
>               asd = kzalloc(sizeof(struct atmel_spi_device), GFP_KERNEL);
>               if (!asd)
>                       return -ENOMEM;
>  
> -             ret = gpio_request(npcs_pin, dev_name(&spi->dev));
> -             if (ret) {
> -                     kfree(asd);
> -                     return ret;
> -             }
> -
> -             asd->npcs_pin = npcs_pin;
>               spi->controller_state = asd;
> -             gpio_direction_output(npcs_pin, !(spi->mode & SPI_CS_HIGH));
>       }
>  
>       asd->csr = csr;
> @@ -1290,6 +1276,15 @@ static int atmel_spi_probe(struct platform_device 
> *pdev)
>       int                     ret;
>       struct spi_master       *master;
>       struct atmel_spi        *as;
> +     struct device_node *np = pdev->dev.of_node;
> +     int num_cs, i;
> +
> +     if (!np)
> +             return -EINVAL;
> +
> +     num_cs = of_gpio_named_count(np, "cs-gpios");
> +     if (num_cs < 0)
> +             return num_cs;
>  

This driver can still be probed without DT, this will break here, please
don't assume that DT is present.


>       /* Select default pin state */
>       pinctrl_pm_select_default_state(&pdev->dev);
> @@ -1308,7 +1303,7 @@ static int atmel_spi_probe(struct platform_device *pdev)
>  
>       /* setup spi core then atmel-specific driver state */
>       ret = -ENOMEM;
> -     master = spi_alloc_master(&pdev->dev, sizeof(*as));
> +     master = spi_alloc_master(&pdev->dev, sizeof(*as) + num_cs * 
> sizeof(unsigned));
>       if (!master)
>               goto out_free;
>  
> @@ -1317,7 +1312,7 @@ static int atmel_spi_probe(struct platform_device *pdev)
>       master->bits_per_word_mask = SPI_BPW_RANGE_MASK(8, 16);
>       master->dev.of_node = pdev->dev.of_node;
>       master->bus_num = pdev->id;
> -     master->num_chipselect = master->dev.of_node ? 0 : 4;
> +     master->num_chipselect = num_cs;

My guess is that you don't need to change that part.

>       master->setup = atmel_spi_setup;
>       master->transfer_one_message = atmel_spi_transfer_one_message;
>       master->cleanup = atmel_spi_cleanup;
> @@ -1325,6 +1320,25 @@ static int atmel_spi_probe(struct platform_device 
> *pdev)
>  
>       as = spi_master_get_devdata(master);
>  
> +     for (i = 0; i < master->num_chipselect; ++i) {
> +             ret = of_get_named_gpio(np, "cs-gpios", i);

Again, DT maybe not be compiled in or that driver may be probed from
platform data.

> +             if (ret < 0) {
> +                     dev_err(&pdev->dev, "failed to get csgpio#%u (%d)\n",
> +                             i, ret);
> +                     goto out_free;
> +             }
> +             as->csgpio[i] = ret;
> +             dev_dbg(&pdev->dev, "csgpio#%u = %u\n", i, as->csgpio[i]);
> +             ret = devm_gpio_request_one(&pdev->dev, as->csgpio[i],
> +                                         GPIOF_OUT_INIT_HIGH, DRIVER_NAME);
> +             if (ret < 0) {
> +                     dev_err(&pdev->dev,
> +                             "failed to configure csgpio#%u (%d)\n",
> +                             i, ret);
> +                     goto out_free;
> +             }
> +     }
> +
>       /*
>        * Scratch buffer is used for throwaway rx and tx data.
>        * It's coherent to minimize dcache pollution.
> -- 
> 1.9.1
> 
> 
> _______________________________________________
> linux-arm-kernel mailing list
> [email protected]
> http://lists.infradead.org/mailman/listinfo/linux-arm-kernel

-- 
Alexandre Belloni, Free Electrons
Embedded Linux, Kernel and Android engineering
http://free-electrons.com
--
To unsubscribe from this list: send the line "unsubscribe devicetree" in
the body of a message to [email protected]
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to