From: <Claudiu.Beznea@microchip.com>
To: <davidm@egauge.net>, <kvalo@codeaurora.org>
Cc: <davem@davemloft.net>, <kuba@kernel.org>, <robh+dt@kernel.org>,
<Ajay.Kathat@microchip.com>, <adham.abozaeid@microchip.com>,
<linux-wireless@vger.kernel.org>, <netdev@vger.kernel.org>,
<devicetree@vger.kernel.org>, <linux-kernel@vger.kernel.org>
Subject: Re: [PATCH] wilc1000: Add reset/enable GPIO support to SPI driver
Date: Tue, 7 Dec 2021 10:11:25 +0000 [thread overview]
Message-ID: <e2a7e005-e133-ec15-df78-2302aa538a26@microchip.com> (raw)
In-Reply-To: <20211202034348.2901092-1-davidm@egauge.net>
On 02.12.2021 05:55, David Mosberger-Tang wrote:
> EXTERNAL EMAIL: Do not click links or open attachments unless you know the content is safe
>
> This patch is based on similar code from the linux4sam/linux-at91 GIT
> repository.
>
> For the SDIO driver, the RESET/ENABLE pins of WILC1000 may be
> controlled through the SDIO power sequence driver. This commit adds
> analogous support for the SPI driver. Specifically, during bus
> probing, the chip will be reset-cycled and during unloading, the chip
> will be placed into reset and disabled (both to reduce power
> consumption and to ensure the WiFi radio is off).
>
> Both RESET and ENABLE GPIOs are optional. However, if the ENABLE GPIO
> is specified, then the RESET GPIO also must be specified as otherwise
> there is no way to ensure proper timing of the ENABLE/RESET sequence.
>
> Signed-off-by: David Mosberger-Tang <davidm@egauge.net>
> ---
> .../net/wireless/microchip,wilc1000.yaml | 11 +++
> .../net/wireless/microchip/wilc1000/Makefile | 2 +-
> drivers/net/wireless/microchip/wilc1000/hif.h | 2 +
> .../net/wireless/microchip/wilc1000/netdev.h | 10 +++
> .../net/wireless/microchip/wilc1000/power.c | 73 +++++++++++++++++++
> drivers/net/wireless/microchip/wilc1000/spi.c | 15 +++-
> .../net/wireless/microchip/wilc1000/wlan.c | 2 +-
> 7 files changed, 110 insertions(+), 5 deletions(-)
> create mode 100644 drivers/net/wireless/microchip/wilc1000/power.c
>
> diff --git a/Documentation/devicetree/bindings/net/wireless/microchip,wilc1000.yaml b/Documentation/devicetree/bindings/net/wireless/microchip,wilc1000.yaml
> index 6c35682377e6..2ce316f5e353 100644
> --- a/Documentation/devicetree/bindings/net/wireless/microchip,wilc1000.yaml
> +++ b/Documentation/devicetree/bindings/net/wireless/microchip,wilc1000.yaml
> @@ -32,6 +32,15 @@ properties:
> clock-names:
> const: rtc
>
> + enable-gpios:
> + maxItems: 1
> + description: A GPIO line connected to the ENABLE line. If this
> + is specified, then reset-gpios also must be specified.
> +
> + reset-gpios:
> + maxItems: 1
> + description: A GPIO line connected to the RESET line.
> +
Bindings should go through a different patch.
> required:
> - compatible
> - interrupts
> @@ -51,6 +60,8 @@ examples:
> interrupts = <27 0>;
> clocks = <&pck1>;
> clock-names = "rtc";
> + enable-gpios = <&pioA 5 0>;
> + reset-gpios = <&pioA 6 0>;
> };
> };
>
> diff --git a/drivers/net/wireless/microchip/wilc1000/Makefile b/drivers/net/wireless/microchip/wilc1000/Makefile
> index c3c9e34c1eaa..baf9f021a1d6 100644
> --- a/drivers/net/wireless/microchip/wilc1000/Makefile
> +++ b/drivers/net/wireless/microchip/wilc1000/Makefile
> @@ -2,7 +2,7 @@
> obj-$(CONFIG_WILC1000) += wilc1000.o
>
> wilc1000-objs := cfg80211.o netdev.o mon.o \
> - hif.o wlan_cfg.o wlan.o
> + hif.o wlan_cfg.o wlan.o power.o
>
> obj-$(CONFIG_WILC1000_SDIO) += wilc1000-sdio.o
> wilc1000-sdio-objs += sdio.o
> diff --git a/drivers/net/wireless/microchip/wilc1000/hif.h b/drivers/net/wireless/microchip/wilc1000/hif.h
> index cccd54ed0518..a57095d8088e 100644
> --- a/drivers/net/wireless/microchip/wilc1000/hif.h
> +++ b/drivers/net/wireless/microchip/wilc1000/hif.h
> @@ -213,4 +213,6 @@ void wilc_network_info_received(struct wilc *wilc, u8 *buffer, u32 length);
> void wilc_gnrl_async_info_received(struct wilc *wilc, u8 *buffer, u32 length);
> void *wilc_parse_join_bss_param(struct cfg80211_bss *bss,
> struct cfg80211_crypto_settings *crypto);
> +int wilc_of_parse_power_pins(struct wilc *wilc);
> +void wilc_wlan_power(struct wilc *wilc, bool on);
> #endif
> diff --git a/drivers/net/wireless/microchip/wilc1000/netdev.h b/drivers/net/wireless/microchip/wilc1000/netdev.h
> index b9a88b3e322f..b95a247322a6 100644
> --- a/drivers/net/wireless/microchip/wilc1000/netdev.h
> +++ b/drivers/net/wireless/microchip/wilc1000/netdev.h
> @@ -197,6 +197,15 @@ struct wilc_vif {
> struct cfg80211_bss *bss;
> };
>
> +struct wilc_power_gpios {
> + int reset;
> + int chip_en;
> +};
> +
> +struct wilc_power {
> + struct wilc_power_gpios gpios;
> +};
> +
> struct wilc_tx_queue_status {
> u8 buffer[AC_BUFFER_SIZE];
> u16 end_index;
> @@ -265,6 +274,7 @@ struct wilc {
> bool suspend_event;
>
> struct workqueue_struct *hif_workqueue;
> + struct wilc_power power;
For SDIO power should be controlled though MMC pwrseq driver. Thus I would
keep this code for SPI only and move this member to struct wilc_spi.
> struct wilc_cfg cfg;
> void *bus_data;
> struct net_device *monitor_dev;
> diff --git a/drivers/net/wireless/microchip/wilc1000/power.c b/drivers/net/wireless/microchip/wilc1000/power.c
> new file mode 100644
> index 000000000000..d26a39b7698d
> --- /dev/null
> +++ b/drivers/net/wireless/microchip/wilc1000/power.c
> @@ -0,0 +1,73 @@
> +// SPDX-License-Identifier: GPL-2.0
> +/*
> + * Copyright (c) 2012 - 2018 Microchip Technology Inc., and its subsidiaries.
> + * All rights reserved.
This is not in the GIT repo. It should have been:
"Copyright (c) 2021 Microchip Technology Inc. and its subsidiaries"
^
current year
> + */
> +#include <linux/delay.h>
> +#include <linux/gpio.h>
> +#include <linux/of.h>
> +#include <linux/of_gpio.h>
> +#include <linux/version.h>
> +
> +#include "netdev.h"
> +
> +/**
> + * wilc_of_parse_power_pins() - parse power sequence pins
> + *
> + * @wilc: wilc data structure
> + *
> + * Returns: 0 on success, negative error number on failures.
> + */
> +int wilc_of_parse_power_pins(struct wilc *wilc)
> +{
> + struct device_node *of = wilc->dev->of_node;
> + struct wilc_power *power = &wilc->power;
> + int ret;
> +
> + power->gpios.reset = of_get_named_gpio_flags(of, "reset-gpios", 0,
> + NULL);
> + power->gpios.chip_en = of_get_named_gpio_flags(of, "chip_en-gpios", 0,
> + NULL);
Consider using gpio descriptors and devm_gpiod_get().
> + if (!gpio_is_valid(power->gpios.reset))
> + return 0; /* assume SDIO power sequence driver is used */
In case of SDIO we should rely on mmc pwrseq all the time. It keeps things
cleaner. I would keep the code in this file only for SPI.
> +
> + if (gpio_is_valid(power->gpios.chip_en)) {
> + ret = devm_gpio_request(wilc->dev, power->gpios.chip_en,
> + "CHIP_EN");
> + if (ret)
> + return ret;
> + }
> + return devm_gpio_request(wilc->dev, power->gpios.reset, "RESET");
> +}
> +EXPORT_SYMBOL_GPL(wilc_of_parse_power_pins);
> +
> +/**
> + * wilc_wlan_power() - handle power on/off commands
> + *
> + * @wilc: wilc data structure
> + * @on: requested power status
> + *
> + * Returns: none
> + */
> +void wilc_wlan_power(struct wilc *wilc, bool on)
> +{
> + if (!gpio_is_valid(wilc->power.gpios.reset)) {
> + /* In case SDIO power sequence driver is used to power this
> + * device then the powering sequence is handled by the bus
> + * via pm_runtime_* functions. */
I see this function is called anyway only in SPI context, so, don't think
this handling for SDIO is necessary. Maybe these is something I miss with
regards to it. Ajay, please correct me if I'm wrong.
> + return;
> + }
> +
> + if (on) {
> + if (gpio_is_valid(wilc->power.gpios.chip_en)) {
> + gpio_direction_output(wilc->power.gpios.chip_en, 1);
> + mdelay(5);
> + }
> + gpio_direction_output(wilc->power.gpios.reset, 1);
> + } else {
> + gpio_direction_output(wilc->power.gpios.reset, 0);
> + if (gpio_is_valid(wilc->power.gpios.chip_en))
> + gpio_direction_output(wilc->power.gpios.chip_en, 0);
> + }
> +}
> +EXPORT_SYMBOL_GPL(wilc_wlan_power);
> diff --git a/drivers/net/wireless/microchip/wilc1000/spi.c b/drivers/net/wireless/microchip/wilc1000/spi.c
> index 640850f989dd..884ad7f954d4 100644
> --- a/drivers/net/wireless/microchip/wilc1000/spi.c
> +++ b/drivers/net/wireless/microchip/wilc1000/spi.c
> @@ -171,6 +171,10 @@ static int wilc_bus_probe(struct spi_device *spi)
> wilc->bus_data = spi_priv;
> wilc->dev_irq_num = spi->irq;
>
> + ret = wilc_of_parse_power_pins(wilc);
> + if (ret)
> + goto netdev_cleanup;
> +
> wilc->rtc_clk = devm_clk_get_optional(&spi->dev, "rtc");
> if (IS_ERR(wilc->rtc_clk)) {
> ret = PTR_ERR(wilc->rtc_clk);
> @@ -178,6 +182,10 @@ static int wilc_bus_probe(struct spi_device *spi)
> }
> clk_prepare_enable(wilc->rtc_clk);
>
> + /* ensure WILC1000 is reset and enabled: */
> + wilc_wlan_power(wilc, false);
> + wilc_wlan_power(wilc, true);
> +
> return 0;
>
> netdev_cleanup:
> @@ -977,9 +985,10 @@ static int wilc_spi_reset(struct wilc *wilc)
>
> static int wilc_spi_deinit(struct wilc *wilc)
> {
> - /*
> - * TODO:
> - */
> + struct wilc_spi *spi_priv = wilc->bus_data;
> +
> + spi_priv->isinit = false;
> + wilc_wlan_power(wilc, false);
> return 0;
> }
>
> diff --git a/drivers/net/wireless/microchip/wilc1000/wlan.c b/drivers/net/wireless/microchip/wilc1000/wlan.c
> index 82566544419a..f1e4ac3a2ad5 100644
> --- a/drivers/net/wireless/microchip/wilc1000/wlan.c
> +++ b/drivers/net/wireless/microchip/wilc1000/wlan.c
> @@ -1253,7 +1253,7 @@ void wilc_wlan_cleanup(struct net_device *dev)
> wilc->rx_buffer = NULL;
> kfree(wilc->tx_buffer);
> wilc->tx_buffer = NULL;
> - wilc->hif_func->hif_deinit(NULL);
> + wilc->hif_func->hif_deinit(wilc);
> }
>
> static int wilc_wlan_cfg_commit(struct wilc_vif *vif, int type,
> --
> 2.25.1
>
next prev parent reply other threads:[~2021-12-07 10:11 UTC|newest]
Thread overview: 5+ messages / expand[flat|nested] mbox.gz Atom feed top
2021-12-02 3:55 [PATCH] wilc1000: Add reset/enable GPIO support to SPI driver David Mosberger-Tang
2021-12-02 15:58 ` Ajay.Kathat
2021-12-07 10:11 ` Claudiu.Beznea [this message]
2021-12-07 12:57 ` Ajay.Kathat
2021-12-08 6:08 ` David Mosberger-Tang
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=e2a7e005-e133-ec15-df78-2302aa538a26@microchip.com \
--to=claudiu.beznea@microchip.com \
--cc=Ajay.Kathat@microchip.com \
--cc=adham.abozaeid@microchip.com \
--cc=davem@davemloft.net \
--cc=davidm@egauge.net \
--cc=devicetree@vger.kernel.org \
--cc=kuba@kernel.org \
--cc=kvalo@codeaurora.org \
--cc=linux-kernel@vger.kernel.org \
--cc=linux-wireless@vger.kernel.org \
--cc=netdev@vger.kernel.org \
--cc=robh+dt@kernel.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 an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.