Commit 0d7b1014 authored by Anton Vorontsov's avatar Anton Vorontsov Committed by Wim Van Sebroeck

[WATCHDOG] mpc8xxx_wdt: add support for MPC8xx watchdogs

The mpc8xxx_wdt driver is using two registers: SWSRR to push magic
numbers, and SWCRR to control the watchdog.  Both registers are available
on the MPC8xx, and seem to have the same offsets and semantics as in
MPC83xx/MPC86xx watchdogs.  The only difference is prescale value.  So
this driver simply works on the MPC8xx CPUs.

One quirk is needed for the MPC8xx, though.  It has small prescale value
and slow CPU, so the watchdog resets board prior to the driver has time to
load.  To solve this we should split initialization in two steps: start
ping the watchdog early, and register the watchdog userspace interface
later.

MPC823 seem to be the first CPU in MPC8xx line, so we use fsl,mpc823-wdt
compatible matching.
Signed-off-by: default avatarAnton Vorontsov <avorontsov@ru.mvista.com>
Tested-by: default avatarJochen Friedrich <jochen@scram.de>
Cc: Kumar Gala <galak@kernel.crashing.org>
Signed-off-by: default avatarWim Van Sebroeck <wim@iguana.be>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
parent cb55d282
...@@ -697,10 +697,11 @@ config 8xx_WDT ...@@ -697,10 +697,11 @@ config 8xx_WDT
config 8xxx_WDT config 8xxx_WDT
tristate "MPC8xxx Platform Watchdog Timer" tristate "MPC8xxx Platform Watchdog Timer"
depends on PPC_83xx || PPC_86xx depends on PPC_8xx || PPC_83xx || PPC_86xx
help help
This driver is for a SoC level watchdog that exists on some This driver is for a SoC level watchdog that exists on some
Freescale PowerPC processors. So far this driver supports: Freescale PowerPC processors. So far this driver supports:
- MPC8xx watchdogs
- MPC83xx watchdogs - MPC83xx watchdogs
- MPC86xx watchdogs - MPC86xx watchdogs
......
/* /*
* mpc8xxx_wdt.c - MPC83xx/MPC86xx watchdog userspace interface * mpc8xxx_wdt.c - MPC8xx/MPC83xx/MPC86xx watchdog userspace interface
* *
* Authors: Dave Updegraff <dave@cray.org> * Authors: Dave Updegraff <dave@cray.org>
* Kumar Gala <galak@kernel.crashing.org> * Kumar Gala <galak@kernel.crashing.org>
...@@ -207,13 +207,6 @@ static int __devinit mpc8xxx_wdt_probe(struct of_device *ofdev, ...@@ -207,13 +207,6 @@ static int __devinit mpc8xxx_wdt_probe(struct of_device *ofdev,
goto err_unmap; goto err_unmap;
} }
ret = misc_register(&mpc8xxx_wdt_miscdev);
if (ret) {
pr_err("cannot register miscdev on minor=%d (err=%d)\n",
WATCHDOG_MINOR, ret);
goto err_unmap;
}
/* Calculate the timeout in seconds */ /* Calculate the timeout in seconds */
if (prescale) if (prescale)
timeout_sec = (timeout * wdt_type->prescaler) / freq; timeout_sec = (timeout * wdt_type->prescaler) / freq;
...@@ -234,6 +227,7 @@ static int __devinit mpc8xxx_wdt_probe(struct of_device *ofdev, ...@@ -234,6 +227,7 @@ static int __devinit mpc8xxx_wdt_probe(struct of_device *ofdev,
return 0; return 0;
err_unmap: err_unmap:
iounmap(wd_base); iounmap(wd_base);
wd_base = NULL;
return ret; return ret;
} }
...@@ -261,6 +255,12 @@ static const struct of_device_id mpc8xxx_wdt_match[] = { ...@@ -261,6 +255,12 @@ static const struct of_device_id mpc8xxx_wdt_match[] = {
.hw_enabled = true, .hw_enabled = true,
}, },
}, },
{
.compatible = "fsl,mpc823-wdt",
.data = &(struct mpc8xxx_wdt_type) {
.prescaler = 0x800,
},
},
{}, {},
}; };
MODULE_DEVICE_TABLE(of, mpc8xxx_wdt_match); MODULE_DEVICE_TABLE(of, mpc8xxx_wdt_match);
...@@ -275,20 +275,42 @@ static struct of_platform_driver mpc8xxx_wdt_driver = { ...@@ -275,20 +275,42 @@ static struct of_platform_driver mpc8xxx_wdt_driver = {
}, },
}; };
/*
* We do wdt initialization in two steps: arch_initcall probes the wdt
* very early to start pinging the watchdog (misc devices are not yet
* available), and later module_init() just registers the misc device.
*/
static int __init mpc8xxx_wdt_init_late(void)
{
int ret;
if (!wd_base)
return -ENODEV;
ret = misc_register(&mpc8xxx_wdt_miscdev);
if (ret) {
pr_err("cannot register miscdev on minor=%d (err=%d)\n",
WATCHDOG_MINOR, ret);
return ret;
}
return 0;
}
module_init(mpc8xxx_wdt_init_late);
static int __init mpc8xxx_wdt_init(void) static int __init mpc8xxx_wdt_init(void)
{ {
return of_register_platform_driver(&mpc8xxx_wdt_driver); return of_register_platform_driver(&mpc8xxx_wdt_driver);
} }
arch_initcall(mpc8xxx_wdt_init);
static void __exit mpc8xxx_wdt_exit(void) static void __exit mpc8xxx_wdt_exit(void)
{ {
of_unregister_platform_driver(&mpc8xxx_wdt_driver); of_unregister_platform_driver(&mpc8xxx_wdt_driver);
} }
subsys_initcall(mpc8xxx_wdt_init);
module_exit(mpc8xxx_wdt_exit); module_exit(mpc8xxx_wdt_exit);
MODULE_AUTHOR("Dave Updegraff, Kumar Gala"); MODULE_AUTHOR("Dave Updegraff, Kumar Gala");
MODULE_DESCRIPTION("Driver for watchdog timer in MPC83xx/MPC86xx uProcessors"); MODULE_DESCRIPTION("Driver for watchdog timer in MPC8xx/MPC83xx/MPC86xx "
"uProcessors");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
MODULE_ALIAS_MISCDEV(WATCHDOG_MINOR); MODULE_ALIAS_MISCDEV(WATCHDOG_MINOR);
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment