aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/wireless/wl12xx/sdio.c
diff options
context:
space:
mode:
authorIdo Yariv <ido@wizery.com>2011-02-27 17:13:58 -0500
committerLuciano Coelho <coelho@ti.com>2011-03-03 09:10:45 -0500
commit11251e7e5c7c5411d1f77dbc7f9bfa2c23626749 (patch)
tree95def2eccf41fbe98b8a4141127b179a466c0d76 /drivers/net/wireless/wl12xx/sdio.c
parentf62c317c1f7f67c249ee9254e55a02fad0d0f86b (diff)
wl12xx: Don't rely on runtime PM for toggling power
Runtime PM might not always be enabled. Even if it is enabled in the running kernel, it can still be temporarily disabled, for instance during suspend. Runtime PM is opportunistic in nature, and should not be relied on for toggling power. In case the interface is removed and re-added while runtime PM is disabled, the FW will fail to boot, as it is mandatory to toggle power between boots. For instance, this can happen during suspend in case one of the devices fails to suspend before the MMC host suspends, but after mac80211 was suspended. The interface will be removed and reactivated without toggling the power. Fix this by calling mmc_power_save_host/mmc_power_restore_host in wl1271_sdio_power_on/off functions. It will toggle the power to the chip even if runtime PM is disabled. The runtime PM functions should still be called to make sure runtime PM does not opportunistically power the chip off (e.g. after resuming from system suspend). Signed-off-by: Ido Yariv <ido@wizery.com> Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com> Signed-off-by: Luciano Coelho <coelho@ti.com>
Diffstat (limited to 'drivers/net/wireless/wl12xx/sdio.c')
-rw-r--r--drivers/net/wireless/wl12xx/sdio.c16
1 files changed, 14 insertions, 2 deletions
diff --git a/drivers/net/wireless/wl12xx/sdio.c b/drivers/net/wireless/wl12xx/sdio.c
index d5e874825069..f27e91502631 100644
--- a/drivers/net/wireless/wl12xx/sdio.c
+++ b/drivers/net/wireless/wl12xx/sdio.c
@@ -28,6 +28,7 @@
28#include <linux/mmc/sdio_func.h> 28#include <linux/mmc/sdio_func.h>
29#include <linux/mmc/sdio_ids.h> 29#include <linux/mmc/sdio_ids.h>
30#include <linux/mmc/card.h> 30#include <linux/mmc/card.h>
31#include <linux/mmc/host.h>
31#include <linux/gpio.h> 32#include <linux/gpio.h>
32#include <linux/wl12xx.h> 33#include <linux/wl12xx.h>
33#include <linux/pm_runtime.h> 34#include <linux/pm_runtime.h>
@@ -163,11 +164,16 @@ static int wl1271_sdio_power_on(struct wl1271 *wl)
163 struct sdio_func *func = wl_to_func(wl); 164 struct sdio_func *func = wl_to_func(wl);
164 int ret; 165 int ret;
165 166
166 /* Power up the card */ 167 /* Make sure the card will not be powered off by runtime PM */
167 ret = pm_runtime_get_sync(&func->dev); 168 ret = pm_runtime_get_sync(&func->dev);
168 if (ret < 0) 169 if (ret < 0)
169 goto out; 170 goto out;
170 171
172 /* Runtime PM might be disabled, so power up the card manually */
173 ret = mmc_power_restore_host(func->card->host);
174 if (ret < 0)
175 goto out;
176
171 sdio_claim_host(func); 177 sdio_claim_host(func);
172 sdio_enable_func(func); 178 sdio_enable_func(func);
173 sdio_release_host(func); 179 sdio_release_host(func);
@@ -179,12 +185,18 @@ out:
179static int wl1271_sdio_power_off(struct wl1271 *wl) 185static int wl1271_sdio_power_off(struct wl1271 *wl)
180{ 186{
181 struct sdio_func *func = wl_to_func(wl); 187 struct sdio_func *func = wl_to_func(wl);
188 int ret;
182 189
183 sdio_claim_host(func); 190 sdio_claim_host(func);
184 sdio_disable_func(func); 191 sdio_disable_func(func);
185 sdio_release_host(func); 192 sdio_release_host(func);
186 193
187 /* Power down the card */ 194 /* Runtime PM might be disabled, so power off the card manually */
195 ret = mmc_power_save_host(func->card->host);
196 if (ret < 0)
197 return ret;
198
199 /* Let runtime PM know the card is powered off */
188 return pm_runtime_put_sync(&func->dev); 200 return pm_runtime_put_sync(&func->dev);
189} 201}
190 202