diff options
| -rw-r--r-- | drivers/watchdog/imx2_wdt.c | 31 |
1 files changed, 22 insertions, 9 deletions
diff --git a/drivers/watchdog/imx2_wdt.c b/drivers/watchdog/imx2_wdt.c index d6add516a7a7..c50c7d85689f 100644 --- a/drivers/watchdog/imx2_wdt.c +++ b/drivers/watchdog/imx2_wdt.c | |||
| @@ -327,18 +327,21 @@ static void imx2_wdt_shutdown(struct platform_device *pdev) | |||
| 327 | } | 327 | } |
| 328 | 328 | ||
| 329 | #ifdef CONFIG_PM_SLEEP | 329 | #ifdef CONFIG_PM_SLEEP |
| 330 | /* Disable watchdog if it is active during suspend */ | 330 | /* Disable watchdog if it is active or non-active but still running */ |
| 331 | static int imx2_wdt_suspend(struct device *dev) | 331 | static int imx2_wdt_suspend(struct device *dev) |
| 332 | { | 332 | { |
| 333 | struct watchdog_device *wdog = dev_get_drvdata(dev); | 333 | struct watchdog_device *wdog = dev_get_drvdata(dev); |
| 334 | struct imx2_wdt_device *wdev = watchdog_get_drvdata(wdog); | 334 | struct imx2_wdt_device *wdev = watchdog_get_drvdata(wdog); |
| 335 | 335 | ||
| 336 | imx2_wdt_set_timeout(wdog, IMX2_WDT_MAX_TIME); | 336 | /* The watchdog IP block is running */ |
| 337 | imx2_wdt_ping(wdog); | 337 | if (imx2_wdt_is_running(wdev)) { |
| 338 | imx2_wdt_set_timeout(wdog, IMX2_WDT_MAX_TIME); | ||
| 339 | imx2_wdt_ping(wdog); | ||
| 338 | 340 | ||
| 339 | /* Watchdog has been stopped but IP block is still running */ | 341 | /* The watchdog is not active */ |
| 340 | if (!watchdog_active(wdog) && imx2_wdt_is_running(wdev)) | 342 | if (!watchdog_active(wdog)) |
| 341 | del_timer_sync(&wdev->timer); | 343 | del_timer_sync(&wdev->timer); |
| 344 | } | ||
| 342 | 345 | ||
| 343 | clk_disable_unprepare(wdev->clk); | 346 | clk_disable_unprepare(wdev->clk); |
| 344 | 347 | ||
| @@ -354,15 +357,25 @@ static int imx2_wdt_resume(struct device *dev) | |||
| 354 | clk_prepare_enable(wdev->clk); | 357 | clk_prepare_enable(wdev->clk); |
| 355 | 358 | ||
| 356 | if (watchdog_active(wdog) && !imx2_wdt_is_running(wdev)) { | 359 | if (watchdog_active(wdog) && !imx2_wdt_is_running(wdev)) { |
| 357 | /* Resumes from deep sleep we need restart | 360 | /* |
| 358 | * the watchdog again. | 361 | * If the watchdog is still active and resumes |
| 362 | * from deep sleep state, need to restart the | ||
| 363 | * watchdog again. | ||
| 359 | */ | 364 | */ |
| 360 | imx2_wdt_setup(wdog); | 365 | imx2_wdt_setup(wdog); |
| 361 | imx2_wdt_set_timeout(wdog, wdog->timeout); | 366 | imx2_wdt_set_timeout(wdog, wdog->timeout); |
| 362 | imx2_wdt_ping(wdog); | 367 | imx2_wdt_ping(wdog); |
| 363 | } else if (imx2_wdt_is_running(wdev)) { | 368 | } else if (imx2_wdt_is_running(wdev)) { |
| 369 | /* Resuming from non-deep sleep state. */ | ||
| 370 | imx2_wdt_set_timeout(wdog, wdog->timeout); | ||
| 364 | imx2_wdt_ping(wdog); | 371 | imx2_wdt_ping(wdog); |
| 365 | mod_timer(&wdev->timer, jiffies + wdog->timeout * HZ / 2); | 372 | /* |
| 373 | * But the watchdog is not active, then start | ||
| 374 | * the timer again. | ||
| 375 | */ | ||
| 376 | if (!watchdog_active(wdog)) | ||
| 377 | mod_timer(&wdev->timer, | ||
| 378 | jiffies + wdog->timeout * HZ / 2); | ||
| 366 | } | 379 | } |
| 367 | 380 | ||
| 368 | return 0; | 381 | return 0; |
