提交 f828d5e2 编写于 作者: L Linus Torvalds

Merge git://www.linux-watchdog.org/linux-watchdog

Pull watchdog fixes from Wim Van Sebroeck:
 "This will fix reboot issues with the imx2_wdt driver and it also drops
  some forgotten owner assignments from platform_drivers"

* git://www.linux-watchdog.org/linux-watchdog:
  watchdog: drop owner assignment from platform_drivers
  watchdog: imx2_wdt: Disable power down counter on boot
  watchdog: imx2_wdt: Improve power management support.
...@@ -503,7 +503,6 @@ static struct platform_driver cdns_wdt_driver = { ...@@ -503,7 +503,6 @@ static struct platform_driver cdns_wdt_driver = {
.shutdown = cdns_wdt_shutdown, .shutdown = cdns_wdt_shutdown,
.driver = { .driver = {
.name = "cdns-wdt", .name = "cdns-wdt",
.owner = THIS_MODULE,
.of_match_table = cdns_wdt_of_match, .of_match_table = cdns_wdt_of_match,
.pm = &cdns_wdt_pm_ops, .pm = &cdns_wdt_pm_ops,
}, },
......
...@@ -52,6 +52,8 @@ ...@@ -52,6 +52,8 @@
#define IMX2_WDT_WRSR 0x04 /* Reset Status Register */ #define IMX2_WDT_WRSR 0x04 /* Reset Status Register */
#define IMX2_WDT_WRSR_TOUT (1 << 1) /* -> Reset due to Timeout */ #define IMX2_WDT_WRSR_TOUT (1 << 1) /* -> Reset due to Timeout */
#define IMX2_WDT_WMCR 0x08 /* Misc Register */
#define IMX2_WDT_MAX_TIME 128 #define IMX2_WDT_MAX_TIME 128
#define IMX2_WDT_DEFAULT_TIME 60 /* in seconds */ #define IMX2_WDT_DEFAULT_TIME 60 /* in seconds */
...@@ -274,6 +276,13 @@ static int __init imx2_wdt_probe(struct platform_device *pdev) ...@@ -274,6 +276,13 @@ static int __init imx2_wdt_probe(struct platform_device *pdev)
imx2_wdt_ping_if_active(wdog); imx2_wdt_ping_if_active(wdog);
/*
* Disable the watchdog power down counter at boot. Otherwise the power
* down counter will pull down the #WDOG interrupt line for one clock
* cycle.
*/
regmap_write(wdev->regmap, IMX2_WDT_WMCR, 0);
ret = watchdog_register_device(wdog); ret = watchdog_register_device(wdog);
if (ret) { if (ret) {
dev_err(&pdev->dev, "cannot register watchdog device\n"); dev_err(&pdev->dev, "cannot register watchdog device\n");
...@@ -327,18 +336,21 @@ static void imx2_wdt_shutdown(struct platform_device *pdev) ...@@ -327,18 +336,21 @@ static void imx2_wdt_shutdown(struct platform_device *pdev)
} }
#ifdef CONFIG_PM_SLEEP #ifdef CONFIG_PM_SLEEP
/* Disable watchdog if it is active during suspend */ /* Disable watchdog if it is active or non-active but still running */
static int imx2_wdt_suspend(struct device *dev) static int imx2_wdt_suspend(struct device *dev)
{ {
struct watchdog_device *wdog = dev_get_drvdata(dev); struct watchdog_device *wdog = dev_get_drvdata(dev);
struct imx2_wdt_device *wdev = watchdog_get_drvdata(wdog); struct imx2_wdt_device *wdev = watchdog_get_drvdata(wdog);
/* The watchdog IP block is running */
if (imx2_wdt_is_running(wdev)) {
imx2_wdt_set_timeout(wdog, IMX2_WDT_MAX_TIME); imx2_wdt_set_timeout(wdog, IMX2_WDT_MAX_TIME);
imx2_wdt_ping(wdog); imx2_wdt_ping(wdog);
/* Watchdog has been stopped but IP block is still running */ /* The watchdog is not active */
if (!watchdog_active(wdog) && imx2_wdt_is_running(wdev)) if (!watchdog_active(wdog))
del_timer_sync(&wdev->timer); del_timer_sync(&wdev->timer);
}
clk_disable_unprepare(wdev->clk); clk_disable_unprepare(wdev->clk);
...@@ -354,15 +366,25 @@ static int imx2_wdt_resume(struct device *dev) ...@@ -354,15 +366,25 @@ static int imx2_wdt_resume(struct device *dev)
clk_prepare_enable(wdev->clk); clk_prepare_enable(wdev->clk);
if (watchdog_active(wdog) && !imx2_wdt_is_running(wdev)) { if (watchdog_active(wdog) && !imx2_wdt_is_running(wdev)) {
/* Resumes from deep sleep we need restart /*
* the watchdog again. * If the watchdog is still active and resumes
* from deep sleep state, need to restart the
* watchdog again.
*/ */
imx2_wdt_setup(wdog); imx2_wdt_setup(wdog);
imx2_wdt_set_timeout(wdog, wdog->timeout); imx2_wdt_set_timeout(wdog, wdog->timeout);
imx2_wdt_ping(wdog); imx2_wdt_ping(wdog);
} else if (imx2_wdt_is_running(wdev)) { } else if (imx2_wdt_is_running(wdev)) {
/* Resuming from non-deep sleep state. */
imx2_wdt_set_timeout(wdog, wdog->timeout);
imx2_wdt_ping(wdog); imx2_wdt_ping(wdog);
mod_timer(&wdev->timer, jiffies + wdog->timeout * HZ / 2); /*
* But the watchdog is not active, then start
* the timer again.
*/
if (!watchdog_active(wdog))
mod_timer(&wdev->timer,
jiffies + wdog->timeout * HZ / 2);
} }
return 0; return 0;
......
...@@ -215,7 +215,6 @@ static struct platform_driver meson_wdt_driver = { ...@@ -215,7 +215,6 @@ static struct platform_driver meson_wdt_driver = {
.remove = meson_wdt_remove, .remove = meson_wdt_remove,
.shutdown = meson_wdt_shutdown, .shutdown = meson_wdt_shutdown,
.driver = { .driver = {
.owner = THIS_MODULE,
.name = DRV_NAME, .name = DRV_NAME,
.of_match_table = meson_wdt_dt_ids, .of_match_table = meson_wdt_dt_ids,
}, },
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册