diff options
Diffstat (limited to 'drivers/watchdog/imx2_wdt.c')
-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; |