Re: [PATCH] wilc1000: Add reset/enable GPIO support to SPI driver

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



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@xxxxxxxxxx>
> ---
>  .../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
> 





[Index of Archives]     [Device Tree Compilter]     [Device Tree Spec]     [Linux Driver Backports]     [Video for Linux]     [Linux USB Devel]     [Linux PCI Devel]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [XFree86]     [Yosemite Backpacking]


  Powered by Linux