From: Alexandre Belloni <alexandre.belloni-wi1+55ScJUtKEb57/3fJTNBPR1lH4CV8@public.gmane.org>
To: Jiri Prchal <jiri.prchal-cKCO0sOKHLPtwjQa/ONI9g@public.gmane.org>
Cc: devicetree-u79uwXL29TY76Z2rM5mHXA@public.gmane.org,
linux-kernel-u79uwXL29TY76Z2rM5mHXA@public.gmane.org,
linux-arm-kernel-IAPFreCvJWM7uuMidbF8XUB+6BGkLq7r@public.gmane.org,
nicolas.ferre-AIFe0yeh4nAAvxtiuMwx3w@public.gmane.org,
voice.shen-AIFe0yeh4nAAvxtiuMwx3w@public.gmane.org
Subject: Re: [PATCH] ARM: at91: spi: request all csgpio in spi probe
Date: Mon, 28 Jul 2014 14:21:03 +0200 [thread overview]
Message-ID: <20140728122103.GR9532@piout.net> (raw)
In-Reply-To: <b033d16ec31d26a895726ee72ec89b43943dfbae.1406547656.git.jiri.prchal-cKCO0sOKHLPtwjQa/ONI9g@public.gmane.org>
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 <jiri.prchal-cKCO0sOKHLPtwjQa/ONI9g@public.gmane.org>
> ---
> 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
> linux-arm-kernel-IAPFreCvJWM7uuMidbF8XUB+6BGkLq7r@public.gmane.org
> 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 majordomo-u79uwXL29TY76Z2rM5mHXA@public.gmane.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
next prev parent reply other threads:[~2014-07-28 12:21 UTC|newest]
Thread overview: 11+ messages / expand[flat|nested] mbox.gz Atom feed top
2014-07-28 11:43 [PATCH] ARM: at91: spi: request all csgpio in spi probe Jiri Prchal
[not found] ` <b033d16ec31d26a895726ee72ec89b43943dfbae.1406547656.git.jiri.prchal-cKCO0sOKHLPtwjQa/ONI9g@public.gmane.org>
2014-07-28 12:21 ` Alexandre Belloni [this message]
2014-07-28 13:06 ` Jiří Prchal
[not found] ` <53D64ADA.1000102-cKCO0sOKHLPtwjQa/ONI9g@public.gmane.org>
2014-07-28 22:38 ` Alexandre Belloni
[not found] ` <20140728223859.GA3214-m++hUPXGwpdeoWH0uzbU5w@public.gmane.org>
2014-07-29 8:00 ` Boris BREZILLON
2014-07-31 15:59 ` Alexandre Belloni
2014-07-31 16:10 ` Jean-Christophe PLAGNIOL-VILLARD
[not found] ` <C5B462AE-B1D2-4067-91EB-44FDF49D28FA-sclMFOaUSTBWk0Htik3J/w@public.gmane.org>
2014-07-31 16:48 ` Alexandre Belloni
2014-07-31 17:05 ` Jean-Christophe PLAGNIOL-VILLARD
2014-07-29 11:32 ` Mark Brown
2014-07-28 15:06 ` Boris BREZILLON
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20140728122103.GR9532@piout.net \
--to=alexandre.belloni-wi1+55scjutkeb57/3fjtnbpr1lh4cv8@public.gmane.org \
--cc=devicetree-u79uwXL29TY76Z2rM5mHXA@public.gmane.org \
--cc=jiri.prchal-cKCO0sOKHLPtwjQa/ONI9g@public.gmane.org \
--cc=linux-arm-kernel-IAPFreCvJWM7uuMidbF8XUB+6BGkLq7r@public.gmane.org \
--cc=linux-kernel-u79uwXL29TY76Z2rM5mHXA@public.gmane.org \
--cc=nicolas.ferre-AIFe0yeh4nAAvxtiuMwx3w@public.gmane.org \
--cc=voice.shen-AIFe0yeh4nAAvxtiuMwx3w@public.gmane.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
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).