Commit 853402a0 authored by Rajat Jain's avatar Rajat Jain Committed by Kalle Valo

mwifiex: Enable WoWLAN for both sdio and pcie

Commit ce4f6f0c ("mwifiex: add platform specific wakeup interrupt
support") added WoWLAN feature only for sdio. This patch moves that
code to the common module so that all the interface drivers can use
it for free. It enables pcie and sdio for its use currently.
Signed-off-by: default avatarRajat Jain <rajatja@google.com>
Signed-off-by: default avatarKalle Valo <kvalo@codeaurora.org>
parent 5e28e5fb
...@@ -1552,14 +1552,55 @@ void mwifiex_do_flr(struct mwifiex_adapter *adapter, bool prepare) ...@@ -1552,14 +1552,55 @@ void mwifiex_do_flr(struct mwifiex_adapter *adapter, bool prepare)
} }
EXPORT_SYMBOL_GPL(mwifiex_do_flr); EXPORT_SYMBOL_GPL(mwifiex_do_flr);
static irqreturn_t mwifiex_irq_wakeup_handler(int irq, void *priv)
{
struct mwifiex_adapter *adapter = priv;
if (adapter->irq_wakeup >= 0) {
dev_dbg(adapter->dev, "%s: wake by wifi", __func__);
adapter->wake_by_wifi = true;
disable_irq_nosync(irq);
}
/* Notify PM core we are wakeup source */
pm_wakeup_event(adapter->dev, 0);
return IRQ_HANDLED;
}
static void mwifiex_probe_of(struct mwifiex_adapter *adapter) static void mwifiex_probe_of(struct mwifiex_adapter *adapter)
{ {
int ret;
struct device *dev = adapter->dev; struct device *dev = adapter->dev;
if (!dev->of_node) if (!dev->of_node)
return; return;
adapter->dt_node = dev->of_node; adapter->dt_node = dev->of_node;
adapter->irq_wakeup = irq_of_parse_and_map(adapter->dt_node, 0);
if (!adapter->irq_wakeup) {
dev_info(dev, "fail to parse irq_wakeup from device tree\n");
return;
}
ret = devm_request_irq(dev, adapter->irq_wakeup,
mwifiex_irq_wakeup_handler, IRQF_TRIGGER_LOW,
"wifi_wake", adapter);
if (ret) {
dev_err(dev, "Failed to request irq_wakeup %d (%d)\n",
adapter->irq_wakeup, ret);
goto err_exit;
}
disable_irq(adapter->irq_wakeup);
if (device_init_wakeup(dev, true)) {
dev_err(dev, "fail to init wakeup for mwifiex\n");
goto err_exit;
}
return;
err_exit:
adapter->irq_wakeup = 0;
} }
/* /*
......
...@@ -1011,6 +1011,10 @@ struct mwifiex_adapter { ...@@ -1011,6 +1011,10 @@ struct mwifiex_adapter {
bool usb_mc_setup; bool usb_mc_setup;
struct cfg80211_wowlan_nd_info *nd_info; struct cfg80211_wowlan_nd_info *nd_info;
struct ieee80211_regdomain *regd; struct ieee80211_regdomain *regd;
/* Wake-on-WLAN (WoWLAN) */
int irq_wakeup;
bool wake_by_wifi;
}; };
void mwifiex_process_tx_queue(struct mwifiex_adapter *adapter); void mwifiex_process_tx_queue(struct mwifiex_adapter *adapter);
...@@ -1410,6 +1414,27 @@ static inline u8 mwifiex_is_tdls_link_setup(u8 status) ...@@ -1410,6 +1414,27 @@ static inline u8 mwifiex_is_tdls_link_setup(u8 status)
return false; return false;
} }
/* Disable platform specific wakeup interrupt */
static inline void mwifiex_disable_wake(struct mwifiex_adapter *adapter)
{
if (adapter->irq_wakeup >= 0) {
disable_irq_wake(adapter->irq_wakeup);
if (!adapter->wake_by_wifi)
disable_irq(adapter->irq_wakeup);
}
}
/* Enable platform specific wakeup interrupt */
static inline void mwifiex_enable_wake(struct mwifiex_adapter *adapter)
{
/* Enable platform specific wakeup interrupt */
if (adapter->irq_wakeup >= 0) {
adapter->wake_by_wifi = false;
enable_irq(adapter->irq_wakeup);
enable_irq_wake(adapter->irq_wakeup);
}
}
int mwifiex_init_shutdown_fw(struct mwifiex_private *priv, int mwifiex_init_shutdown_fw(struct mwifiex_private *priv,
u32 func_init_shutdown); u32 func_init_shutdown);
int mwifiex_add_card(void *card, struct semaphore *sem, int mwifiex_add_card(void *card, struct semaphore *sem,
......
...@@ -131,6 +131,7 @@ static int mwifiex_pcie_suspend(struct device *dev) ...@@ -131,6 +131,7 @@ static int mwifiex_pcie_suspend(struct device *dev)
} }
adapter = card->adapter; adapter = card->adapter;
mwifiex_enable_wake(adapter);
/* Enable the Host Sleep */ /* Enable the Host Sleep */
if (!mwifiex_enable_hs(adapter)) { if (!mwifiex_enable_hs(adapter)) {
...@@ -186,6 +187,7 @@ static int mwifiex_pcie_resume(struct device *dev) ...@@ -186,6 +187,7 @@ static int mwifiex_pcie_resume(struct device *dev)
mwifiex_cancel_hs(mwifiex_get_priv(adapter, MWIFIEX_BSS_ROLE_STA), mwifiex_cancel_hs(mwifiex_get_priv(adapter, MWIFIEX_BSS_ROLE_STA),
MWIFIEX_ASYNC_CMD); MWIFIEX_ASYNC_CMD);
mwifiex_disable_wake(adapter);
return 0; return 0;
} }
......
...@@ -79,67 +79,18 @@ static const struct of_device_id mwifiex_sdio_of_match_table[] = { ...@@ -79,67 +79,18 @@ static const struct of_device_id mwifiex_sdio_of_match_table[] = {
{ } { }
}; };
static irqreturn_t mwifiex_wake_irq_wifi(int irq, void *priv)
{
struct mwifiex_plt_wake_cfg *cfg = priv;
if (cfg->irq_wifi >= 0) {
pr_info("%s: wake by wifi", __func__);
cfg->wake_by_wifi = true;
disable_irq_nosync(irq);
}
/* Notify PM core we are wakeup source */
pm_wakeup_event(cfg->dev, 0);
return IRQ_HANDLED;
}
/* This function parse device tree node using mmc subnode devicetree API. /* This function parse device tree node using mmc subnode devicetree API.
* The device node is saved in card->plt_of_node. * The device node is saved in card->plt_of_node.
* if the device tree node exist and include interrupts attributes, this * if the device tree node exist and include interrupts attributes, this
* function will also request platform specific wakeup interrupt. * function will also request platform specific wakeup interrupt.
*/ */
static int mwifiex_sdio_probe_of(struct device *dev, struct sdio_mmc_card *card) static int mwifiex_sdio_probe_of(struct device *dev)
{ {
struct mwifiex_plt_wake_cfg *cfg;
int ret;
if (!of_match_node(mwifiex_sdio_of_match_table, dev->of_node)) { if (!of_match_node(mwifiex_sdio_of_match_table, dev->of_node)) {
dev_err(dev, "required compatible string missing\n"); dev_err(dev, "required compatible string missing\n");
return -EINVAL; return -EINVAL;
} }
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->dev = dev;
cfg->irq_wifi = irq_of_parse_and_map(card->plt_of_node, 0);
if (!cfg->irq_wifi) {
dev_dbg(dev,
"fail to parse irq_wifi from device tree\n");
} else {
ret = devm_request_irq(dev, cfg->irq_wifi,
mwifiex_wake_irq_wifi,
IRQF_TRIGGER_LOW,
"wifi_wake", cfg);
if (ret) {
dev_dbg(dev,
"Failed to request irq_wifi %d (%d)\n",
cfg->irq_wifi, ret);
card->plt_wake_cfg = NULL;
return 0;
}
disable_irq(cfg->irq_wifi);
}
}
ret = device_init_wakeup(dev, true);
if (ret)
dev_err(dev, "fail to init wakeup for mwifiex");
return 0; return 0;
} }
...@@ -198,12 +149,10 @@ mwifiex_sdio_probe(struct sdio_func *func, const struct sdio_device_id *id) ...@@ -198,12 +149,10 @@ mwifiex_sdio_probe(struct sdio_func *func, const struct sdio_device_id *id)
/* device tree node parsing and platform specific configuration*/ /* device tree node parsing and platform specific configuration*/
if (func->dev.of_node) { if (func->dev.of_node) {
ret = mwifiex_sdio_probe_of(&func->dev, card); ret = mwifiex_sdio_probe_of(&func->dev);
if (ret) { if (ret)
dev_err(&func->dev, "SDIO dt node parse failed\n");
goto err_disable; goto err_disable;
} }
}
ret = mwifiex_add_card(card, &add_remove_card_sem, &sdio_ops, ret = mwifiex_add_card(card, &add_remove_card_sem, &sdio_ops,
MWIFIEX_SDIO, &func->dev); MWIFIEX_SDIO, &func->dev);
...@@ -265,12 +214,7 @@ static int mwifiex_sdio_resume(struct device *dev) ...@@ -265,12 +214,7 @@ static int mwifiex_sdio_resume(struct device *dev)
mwifiex_cancel_hs(mwifiex_get_priv(adapter, MWIFIEX_BSS_ROLE_STA), mwifiex_cancel_hs(mwifiex_get_priv(adapter, MWIFIEX_BSS_ROLE_STA),
MWIFIEX_SYNC_CMD); MWIFIEX_SYNC_CMD);
/* Disable platform specific wakeup interrupt */ mwifiex_disable_wake(adapter);
if (card->plt_wake_cfg && card->plt_wake_cfg->irq_wifi >= 0) {
disable_irq_wake(card->plt_wake_cfg->irq_wifi);
if (!card->plt_wake_cfg->wake_by_wifi)
disable_irq(card->plt_wake_cfg->irq_wifi);
}
return 0; return 0;
} }
...@@ -350,13 +294,7 @@ static int mwifiex_sdio_suspend(struct device *dev) ...@@ -350,13 +294,7 @@ static int mwifiex_sdio_suspend(struct device *dev)
} }
adapter = card->adapter; adapter = card->adapter;
mwifiex_enable_wake(adapter);
/* Enable platform specific wakeup interrupt */
if (card->plt_wake_cfg && card->plt_wake_cfg->irq_wifi >= 0) {
card->plt_wake_cfg->wake_by_wifi = false;
enable_irq(card->plt_wake_cfg->irq_wifi);
enable_irq_wake(card->plt_wake_cfg->irq_wifi);
}
/* Enable the Host Sleep */ /* Enable the Host Sleep */
if (!mwifiex_enable_hs(adapter)) { if (!mwifiex_enable_hs(adapter)) {
......
...@@ -154,12 +154,6 @@ ...@@ -154,12 +154,6 @@
a->mpa_rx.start_port = 0; \ a->mpa_rx.start_port = 0; \
} while (0) } while (0)
struct mwifiex_plt_wake_cfg {
struct device *dev;
int irq_wifi;
bool wake_by_wifi;
};
/* data structure for SDIO MPA TX */ /* data structure for SDIO MPA TX */
struct mwifiex_sdio_mpa_tx { struct mwifiex_sdio_mpa_tx {
/* multiport tx aggregation buffer pointer */ /* multiport tx aggregation buffer pointer */
...@@ -243,8 +237,6 @@ struct mwifiex_sdio_card_reg { ...@@ -243,8 +237,6 @@ struct mwifiex_sdio_card_reg {
struct sdio_mmc_card { struct sdio_mmc_card {
struct sdio_func *func; struct sdio_func *func;
struct mwifiex_adapter *adapter; struct mwifiex_adapter *adapter;
struct device_node *plt_of_node;
struct mwifiex_plt_wake_cfg *plt_wake_cfg;
const char *firmware; const char *firmware;
const struct mwifiex_sdio_card_reg *reg; const struct mwifiex_sdio_card_reg *reg;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment