RE: [PATCH 2/2] btmrvl: add platform specific wakeup interrupt support
From: Amitkumar Karwar
Date: Fri Mar 18 2016 - 07:14:14 EST
> From: Amitkumar Karwar [mailto:akarwar@xxxxxxxxxxx]
> Sent: Friday, March 18, 2016 4:35 PM
> To: linux-bluetooth@xxxxxxxxxxxxxxx
> Cc: Cathy Luo; linux-kernel@xxxxxxxxxxxxxxx; Nishant Sarmukadam;
> devicetree@xxxxxxxxxxxxxxx; Wei-Ning Huang; Xinming Hu; Amitkumar Karwar
> Subject: [PATCH 2/2] btmrvl: add platform specific wakeup interrupt
> support
>
> From: Xinming Hu <huxm@xxxxxxxxxxx>
>
> On some arm-based platforms, we need to configure platform specific
> parameters by device tree node and we need define our node as a child
> node of parent SDIO host controller.
> This patch parses these parameters from device tree. It includes
> calibration data download to firmware, wakeup pin configured to
> firmware, and soc specific wakeup interrupt pin.
>
> Signed-off-by: Xinming Hu <huxm@xxxxxxxxxxx>
> Signed-off-by: Amitkumar Karwar <akarwar@xxxxxxxxxxx>
> ---
> drivers/bluetooth/btmrvl_drv.h | 11 ++++++
> drivers/bluetooth/btmrvl_main.c | 20 +++++------
> drivers/bluetooth/btmrvl_sdio.c | 79
> +++++++++++++++++++++++++++++++++++++++++
> drivers/bluetooth/btmrvl_sdio.h | 6 ++++
> 4 files changed, 105 insertions(+), 11 deletions(-)
>
> diff --git a/drivers/bluetooth/btmrvl_drv.h
> b/drivers/bluetooth/btmrvl_drv.h index 0590473..f742384 100644
> --- a/drivers/bluetooth/btmrvl_drv.h
> +++ b/drivers/bluetooth/btmrvl_drv.h
> @@ -23,6 +23,17 @@
> #include <linux/bitops.h>
> #include <linux/slab.h>
> #include <net/bluetooth/bluetooth.h>
> +#include <linux/err.h>
> +#include <linux/gpio.h>
> +#include <linux/gfp.h>
> +#include <linux/interrupt.h>
> +#include <linux/io.h>
> +#include <linux/of_gpio.h>
> +#include <linux/of_platform.h>
> +#include <linux/platform_device.h>
> +#include <linux/pm_runtime.h>
> +#include <linux/slab.h>
> +#include <linux/of_irq.h>
>
> #define BTM_HEADER_LEN 4
> #define BTM_UPLD_SIZE 2312
> diff --git a/drivers/bluetooth/btmrvl_main.c
> b/drivers/bluetooth/btmrvl_main.c index f25a825..b7c32f4 100644
> --- a/drivers/bluetooth/btmrvl_main.c
> +++ b/drivers/bluetooth/btmrvl_main.c
> @@ -510,34 +510,32 @@ static int btmrvl_download_cal_data(struct
> btmrvl_private *priv, static int btmrvl_check_device_tree(struct
> btmrvl_private *priv) {
> struct device_node *dt_node;
> + struct btmrvl_sdio_card *card = priv->btmrvl_dev.card;
> u8 cal_data[BT_CAL_HDR_LEN + BT_CAL_DATA_SIZE];
> - int ret;
> + int ret = 0;
> u32 val;
>
> - for_each_compatible_node(dt_node, NULL, "btmrvl,cfgdata") {
> - ret = of_property_read_u32(dt_node, "btmrvl,gpio-gap",
> &val);
> + if (card->plt_of_node) {
> + dt_node = card->plt_of_node;
> + ret = of_property_read_u32(dt_node, "btmrvl,wakeuppin-gap",
> + &val);
> if (!ret)
> priv->btmrvl_dev.gpio_gap = val;
>
> ret = of_property_read_u8_array(dt_node, "btmrvl,cal-data",
> cal_data + BT_CAL_HDR_LEN,
> BT_CAL_DATA_SIZE);
> - if (ret) {
> - of_node_put(dt_node);
> + if (ret)
> return ret;
> - }
>
> BT_DBG("Use cal data from device tree");
> ret = btmrvl_download_cal_data(priv, cal_data,
> BT_CAL_DATA_SIZE);
> - if (ret) {
> + if (ret)
> BT_ERR("Fail to download calibrate data");
> - of_node_put(dt_node);
> - return ret;
> - }
> }
>
> - return 0;
> + return ret;
> }
>
> static int btmrvl_setup(struct hci_dev *hdev) diff --git
> a/drivers/bluetooth/btmrvl_sdio.c b/drivers/bluetooth/btmrvl_sdio.c
> index 6ed8acf..845ef28 100644
> --- a/drivers/bluetooth/btmrvl_sdio.c
> +++ b/drivers/bluetooth/btmrvl_sdio.c
> @@ -52,6 +52,68 @@ static struct memory_type_mapping
> mem_type_mapping_tbl[] = {
> {"EXTLAST", NULL, 0, 0xFE},
> };
>
> +static const struct of_device_id btmrvl_sdio_of_match_table[] = {
> + { .compatible = "marvell,bt-sd8897" },
> + { .compatible = "marvell,bt-sd8997" },
> + { }
> +};
> +
> +static irqreturn_t btmrvl_wake_irq_bt(int irq, void *priv) {
> + struct btmrvl_plt_wake_cfg *cfg = priv;
> +
> + if (cfg->irq_bt >= 0) {
> + pr_info("%s: wake by bt", __func__);
> + cfg->wake_by_bt = true;
> + disable_irq_nosync(irq);
> + }
> +
> + return IRQ_HANDLED;
> +}
> +
> +/* This function parse device tree node using mmc subnode devicetree
> API.
> + * The device node is saved in card->plt_of_node.
> + * if the device tree node exist and include interrupts attributes,
> +this
> + * function will also request platform specific wakeup interrupt.
> + */
> +static int btmrvl_sdio_probe_of(struct device *dev,
> + struct btmrvl_sdio_card *card)
> +{
> + struct btmrvl_plt_wake_cfg *cfg;
> + int ret;
> +
> + if (!dev->of_node ||
> + !of_match_node(btmrvl_sdio_of_match_table, dev->of_node)) {
> + pr_err("sdio platform data not available");
> + return -1;
> + }
> +
> + card->plt_of_node = dev->of_node;
> +
> + card->plt_wake_cfg = devm_kzalloc(dev, sizeof(*card-
> >plt_wake_cfg),
> + GFP_KERNEL);
> + cfg = card->plt_wake_cfg;
> + if (cfg && card->plt_of_node) {
> + cfg->irq_bt = irq_of_parse_and_map(card->plt_of_node, 0);
> + if (!cfg->irq_bt) {
> + dev_err(dev, "fail to parse irq_bt from device tree");
> + } else {
> + ret = devm_request_irq(dev, cfg->irq_bt,
> + btmrvl_wake_irq_bt,
> + IRQF_TRIGGER_LOW,
> + "bt_wake", cfg);
> + if (ret) {
> + dev_err(dev,
> + "Failed to request irq_bt %d (%d)\n",
> + cfg->irq_bt, ret);
> + }
> + disable_irq(cfg->irq_bt);
> + }
> + }
> +
> + return 0;
> +}
> +
> /* The btmrvl_sdio_remove() callback function is called
> * when user removes this module from kernel space or ejects
> * the card from the slot. The driver handles these 2 cases @@ -1464,6
> +1526,9 @@ static int btmrvl_sdio_probe(struct sdio_func *func,
>
> btmrvl_sdio_enable_host_int(card);
>
> + /* device tree node parsing and platform specific configuration*/
> + btmrvl_sdio_probe_of(&func->dev, card);
> +
> priv = btmrvl_add_card(card);
> if (!priv) {
> BT_ERR("Initializing card failed!");
> @@ -1544,6 +1609,13 @@ static int btmrvl_sdio_suspend(struct device
> *dev)
> return 0;
> }
>
> + /* Enable platform specific wakeup interrupt */
> + if (card->plt_wake_cfg && card->plt_wake_cfg->irq_bt >= 0) {
> + card->plt_wake_cfg->wake_by_bt = false;
> + enable_irq(card->plt_wake_cfg->irq_bt);
> + enable_irq_wake(card->plt_wake_cfg->irq_bt);
> + }
> +
> priv = card->priv;
> priv->adapter->is_suspending = true;
> hcidev = priv->btmrvl_dev.hcidev;
> @@ -1606,6 +1678,13 @@ static int btmrvl_sdio_resume(struct device *dev)
> BT_DBG("%s: SDIO resume", hcidev->name);
> hci_resume_dev(hcidev);
>
> + /* Disable platform specific wakeup interrupt */
> + if (card->plt_wake_cfg && card->plt_wake_cfg->irq_bt >= 0) {
> + disable_irq_wake(card->plt_wake_cfg->irq_bt);
> + if (!card->plt_wake_cfg->wake_by_bt)
> + disable_irq(card->plt_wake_cfg->irq_bt);
> + }
> +
> return 0;
> }
>
> diff --git a/drivers/bluetooth/btmrvl_sdio.h
> b/drivers/bluetooth/btmrvl_sdio.h index 1a3bd06..3a522d2 100644
> --- a/drivers/bluetooth/btmrvl_sdio.h
> +++ b/drivers/bluetooth/btmrvl_sdio.h
> @@ -62,6 +62,10 @@
>
> #define FIRMWARE_READY 0xfedc
>
> +struct btmrvl_plt_wake_cfg {
> + int irq_bt;
> + bool wake_by_bt;
> +};
>
> struct btmrvl_sdio_card_reg {
> u8 cfg;
> @@ -97,6 +101,8 @@ struct btmrvl_sdio_card {
> u16 sd_blksz_fw_dl;
> u8 rx_unit;
> struct btmrvl_private *priv;
> + struct device_node *plt_of_node;
> + struct btmrvl_plt_wake_cfg *plt_wake_cfg;
> };
>
> struct btmrvl_sdio_device {
> --
> 1.8.1.4
Kindly ignore this patch. I missed to edit the version. i.e. PATCH 2/2 -> PATCH v5 2/2
It has been resent.
Regards,
Amit