aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAlexandre Belloni <alexandre.belloni@free-electrons.com>2015-08-16 05:23:43 -0400
committerWim Van Sebroeck <wim@iguana.be>2015-09-09 15:38:52 -0400
commita97a09bd119fbdf4ba8c634fed8f4148d1def1e0 (patch)
tree90af60d8025852168cf758f7b4f9499e058339aa
parentf4fff94e3e3a712ef062c44b64ecf8f552f48ea4 (diff)
watchdog: at91sam9: get and use slow clock
Commit dca1a4b5ff6e ("clk: at91: keep slow clk enabled to prevent system hang") added a workaround for the slow clock as it is not properly handled by its users. Get and use the slow clock as it is necessary for the at91sam9 watchdog. Signed-off-by: Alexandre Belloni <alexandre.belloni@free-electrons.com> Reviewed-by: Guenter Roeck <linux@roeck-us.net> Signed-off-by: Wim Van Sebroeck <wim@iguana.be>
-rw-r--r--drivers/watchdog/at91sam9_wdt.c22
1 files changed, 20 insertions, 2 deletions
diff --git a/drivers/watchdog/at91sam9_wdt.c b/drivers/watchdog/at91sam9_wdt.c
index e4698f7c5f93..7e6acaf3ece4 100644
--- a/drivers/watchdog/at91sam9_wdt.c
+++ b/drivers/watchdog/at91sam9_wdt.c
@@ -17,6 +17,7 @@
17 17
18#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt 18#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
19 19
20#include <linux/clk.h>
20#include <linux/errno.h> 21#include <linux/errno.h>
21#include <linux/init.h> 22#include <linux/init.h>
22#include <linux/interrupt.h> 23#include <linux/interrupt.h>
@@ -90,6 +91,7 @@ struct at91wdt {
90 unsigned long heartbeat; /* WDT heartbeat in jiffies */ 91 unsigned long heartbeat; /* WDT heartbeat in jiffies */
91 bool nowayout; 92 bool nowayout;
92 unsigned int irq; 93 unsigned int irq;
94 struct clk *sclk;
93}; 95};
94 96
95/* ......................................................................... */ 97/* ......................................................................... */
@@ -352,15 +354,25 @@ static int __init at91wdt_probe(struct platform_device *pdev)
352 if (IS_ERR(wdt->base)) 354 if (IS_ERR(wdt->base))
353 return PTR_ERR(wdt->base); 355 return PTR_ERR(wdt->base);
354 356
357 wdt->sclk = devm_clk_get(&pdev->dev, NULL);
358 if (IS_ERR(wdt->sclk))
359 return PTR_ERR(wdt->sclk);
360
361 err = clk_prepare_enable(wdt->sclk);
362 if (err) {
363 dev_err(&pdev->dev, "Could not enable slow clock\n");
364 return err;
365 }
366
355 if (pdev->dev.of_node) { 367 if (pdev->dev.of_node) {
356 err = of_at91wdt_init(pdev->dev.of_node, wdt); 368 err = of_at91wdt_init(pdev->dev.of_node, wdt);
357 if (err) 369 if (err)
358 return err; 370 goto err_clk;
359 } 371 }
360 372
361 err = at91_wdt_init(pdev, wdt); 373 err = at91_wdt_init(pdev, wdt);
362 if (err) 374 if (err)
363 return err; 375 goto err_clk;
364 376
365 platform_set_drvdata(pdev, wdt); 377 platform_set_drvdata(pdev, wdt);
366 378
@@ -368,6 +380,11 @@ static int __init at91wdt_probe(struct platform_device *pdev)
368 wdt->wdd.timeout, wdt->nowayout); 380 wdt->wdd.timeout, wdt->nowayout);
369 381
370 return 0; 382 return 0;
383
384err_clk:
385 clk_disable_unprepare(wdt->sclk);
386
387 return err;
371} 388}
372 389
373static int __exit at91wdt_remove(struct platform_device *pdev) 390static int __exit at91wdt_remove(struct platform_device *pdev)
@@ -377,6 +394,7 @@ static int __exit at91wdt_remove(struct platform_device *pdev)
377 394
378 pr_warn("I quit now, hardware will probably reboot!\n"); 395 pr_warn("I quit now, hardware will probably reboot!\n");
379 del_timer(&wdt->timer); 396 del_timer(&wdt->timer);
397 clk_disable_unprepare(wdt->sclk);
380 398
381 return 0; 399 return 0;
382} 400}