|  | /* | 
|  | * PIC32 deadman timer driver | 
|  | * | 
|  | * Purna Chandra Mandal <purna.mandal@microchip.com> | 
|  | * Copyright (c) 2016, Microchip Technology Inc. | 
|  | * | 
|  | * This program is free software; you can redistribute it and/or | 
|  | * modify it under the terms of the GNU General Public License | 
|  | * as published by the Free Software Foundation; either version | 
|  | * 2 of the License, or (at your option) any later version. | 
|  | */ | 
|  | #include <linux/clk.h> | 
|  | #include <linux/device.h> | 
|  | #include <linux/err.h> | 
|  | #include <linux/io.h> | 
|  | #include <linux/kernel.h> | 
|  | #include <linux/module.h> | 
|  | #include <linux/of.h> | 
|  | #include <linux/of_device.h> | 
|  | #include <linux/platform_device.h> | 
|  | #include <linux/pm.h> | 
|  | #include <linux/watchdog.h> | 
|  |  | 
|  | #include <asm/mach-pic32/pic32.h> | 
|  |  | 
|  | /* Deadman Timer Regs */ | 
|  | #define DMTCON_REG	0x00 | 
|  | #define DMTPRECLR_REG	0x10 | 
|  | #define DMTCLR_REG	0x20 | 
|  | #define DMTSTAT_REG	0x30 | 
|  | #define DMTCNT_REG	0x40 | 
|  | #define DMTPSCNT_REG	0x60 | 
|  | #define DMTPSINTV_REG	0x70 | 
|  |  | 
|  | /* Deadman Timer Regs fields */ | 
|  | #define DMT_ON			BIT(15) | 
|  | #define DMT_STEP1_KEY		BIT(6) | 
|  | #define DMT_STEP2_KEY		BIT(3) | 
|  | #define DMTSTAT_WINOPN		BIT(0) | 
|  | #define DMTSTAT_EVENT		BIT(5) | 
|  | #define DMTSTAT_BAD2		BIT(6) | 
|  | #define DMTSTAT_BAD1		BIT(7) | 
|  |  | 
|  | /* Reset Control Register fields for watchdog */ | 
|  | #define RESETCON_DMT_TIMEOUT	BIT(5) | 
|  |  | 
|  | struct pic32_dmt { | 
|  | void __iomem	*regs; | 
|  | struct clk	*clk; | 
|  | }; | 
|  |  | 
|  | static inline void dmt_enable(struct pic32_dmt *dmt) | 
|  | { | 
|  | writel(DMT_ON, PIC32_SET(dmt->regs + DMTCON_REG)); | 
|  | } | 
|  |  | 
|  | static inline void dmt_disable(struct pic32_dmt *dmt) | 
|  | { | 
|  | writel(DMT_ON, PIC32_CLR(dmt->regs + DMTCON_REG)); | 
|  | /* | 
|  | * Cannot touch registers in the CPU cycle following clearing the | 
|  | * ON bit. | 
|  | */ | 
|  | nop(); | 
|  | } | 
|  |  | 
|  | static inline int dmt_bad_status(struct pic32_dmt *dmt) | 
|  | { | 
|  | u32 val; | 
|  |  | 
|  | val = readl(dmt->regs + DMTSTAT_REG); | 
|  | val &= (DMTSTAT_BAD1 | DMTSTAT_BAD2 | DMTSTAT_EVENT); | 
|  | if (val) | 
|  | return -EAGAIN; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static inline int dmt_keepalive(struct pic32_dmt *dmt) | 
|  | { | 
|  | u32 v; | 
|  | u32 timeout = 500; | 
|  |  | 
|  | /* set pre-clear key */ | 
|  | writel(DMT_STEP1_KEY << 8, dmt->regs + DMTPRECLR_REG); | 
|  |  | 
|  | /* wait for DMT window to open */ | 
|  | while (--timeout) { | 
|  | v = readl(dmt->regs + DMTSTAT_REG) & DMTSTAT_WINOPN; | 
|  | if (v == DMTSTAT_WINOPN) | 
|  | break; | 
|  | } | 
|  |  | 
|  | /* apply key2 */ | 
|  | writel(DMT_STEP2_KEY, dmt->regs + DMTCLR_REG); | 
|  |  | 
|  | /* check whether keys are latched correctly */ | 
|  | return dmt_bad_status(dmt); | 
|  | } | 
|  |  | 
|  | static inline u32 pic32_dmt_get_timeout_secs(struct pic32_dmt *dmt) | 
|  | { | 
|  | unsigned long rate; | 
|  |  | 
|  | rate = clk_get_rate(dmt->clk); | 
|  | if (rate) | 
|  | return readl(dmt->regs + DMTPSCNT_REG) / rate; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static inline u32 pic32_dmt_bootstatus(struct pic32_dmt *dmt) | 
|  | { | 
|  | u32 v; | 
|  | void __iomem *rst_base; | 
|  |  | 
|  | rst_base = ioremap(PIC32_BASE_RESET, 0x10); | 
|  | if (!rst_base) | 
|  | return 0; | 
|  |  | 
|  | v = readl(rst_base); | 
|  |  | 
|  | writel(RESETCON_DMT_TIMEOUT, PIC32_CLR(rst_base)); | 
|  |  | 
|  | iounmap(rst_base); | 
|  | return v & RESETCON_DMT_TIMEOUT; | 
|  | } | 
|  |  | 
|  | static int pic32_dmt_start(struct watchdog_device *wdd) | 
|  | { | 
|  | struct pic32_dmt *dmt = watchdog_get_drvdata(wdd); | 
|  |  | 
|  | dmt_enable(dmt); | 
|  | return dmt_keepalive(dmt); | 
|  | } | 
|  |  | 
|  | static int pic32_dmt_stop(struct watchdog_device *wdd) | 
|  | { | 
|  | struct pic32_dmt *dmt = watchdog_get_drvdata(wdd); | 
|  |  | 
|  | dmt_disable(dmt); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int pic32_dmt_ping(struct watchdog_device *wdd) | 
|  | { | 
|  | struct pic32_dmt *dmt = watchdog_get_drvdata(wdd); | 
|  |  | 
|  | return dmt_keepalive(dmt); | 
|  | } | 
|  |  | 
|  | static const struct watchdog_ops pic32_dmt_fops = { | 
|  | .owner		= THIS_MODULE, | 
|  | .start		= pic32_dmt_start, | 
|  | .stop		= pic32_dmt_stop, | 
|  | .ping		= pic32_dmt_ping, | 
|  | }; | 
|  |  | 
|  | static const struct watchdog_info pic32_dmt_ident = { | 
|  | .options	= WDIOF_KEEPALIVEPING | | 
|  | WDIOF_MAGICCLOSE, | 
|  | .identity	= "PIC32 Deadman Timer", | 
|  | }; | 
|  |  | 
|  | static struct watchdog_device pic32_dmt_wdd = { | 
|  | .info		= &pic32_dmt_ident, | 
|  | .ops		= &pic32_dmt_fops, | 
|  | }; | 
|  |  | 
|  | static int pic32_dmt_probe(struct platform_device *pdev) | 
|  | { | 
|  | int ret; | 
|  | struct pic32_dmt *dmt; | 
|  | struct resource *mem; | 
|  | struct watchdog_device *wdd = &pic32_dmt_wdd; | 
|  |  | 
|  | dmt = devm_kzalloc(&pdev->dev, sizeof(*dmt), GFP_KERNEL); | 
|  | if (!dmt) | 
|  | return -ENOMEM; | 
|  |  | 
|  | mem = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 
|  | dmt->regs = devm_ioremap_resource(&pdev->dev, mem); | 
|  | if (IS_ERR(dmt->regs)) | 
|  | return PTR_ERR(dmt->regs); | 
|  |  | 
|  | dmt->clk = devm_clk_get(&pdev->dev, NULL); | 
|  | if (IS_ERR(dmt->clk)) { | 
|  | dev_err(&pdev->dev, "clk not found\n"); | 
|  | return PTR_ERR(dmt->clk); | 
|  | } | 
|  |  | 
|  | ret = clk_prepare_enable(dmt->clk); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | wdd->timeout = pic32_dmt_get_timeout_secs(dmt); | 
|  | if (!wdd->timeout) { | 
|  | dev_err(&pdev->dev, | 
|  | "failed to read watchdog register timeout\n"); | 
|  | ret = -EINVAL; | 
|  | goto out_disable_clk; | 
|  | } | 
|  |  | 
|  | dev_info(&pdev->dev, "timeout %d\n", wdd->timeout); | 
|  |  | 
|  | wdd->bootstatus = pic32_dmt_bootstatus(dmt) ? WDIOF_CARDRESET : 0; | 
|  |  | 
|  | watchdog_set_nowayout(wdd, WATCHDOG_NOWAYOUT); | 
|  | watchdog_set_drvdata(wdd, dmt); | 
|  |  | 
|  | ret = watchdog_register_device(wdd); | 
|  | if (ret) { | 
|  | dev_err(&pdev->dev, "watchdog register failed, err %d\n", ret); | 
|  | goto out_disable_clk; | 
|  | } | 
|  |  | 
|  | platform_set_drvdata(pdev, wdd); | 
|  | return 0; | 
|  |  | 
|  | out_disable_clk: | 
|  | clk_disable_unprepare(dmt->clk); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int pic32_dmt_remove(struct platform_device *pdev) | 
|  | { | 
|  | struct watchdog_device *wdd = platform_get_drvdata(pdev); | 
|  | struct pic32_dmt *dmt = watchdog_get_drvdata(wdd); | 
|  |  | 
|  | watchdog_unregister_device(wdd); | 
|  | clk_disable_unprepare(dmt->clk); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static const struct of_device_id pic32_dmt_of_ids[] = { | 
|  | { .compatible = "microchip,pic32mzda-dmt",}, | 
|  | { /* sentinel */ } | 
|  | }; | 
|  | MODULE_DEVICE_TABLE(of, pic32_dmt_of_ids); | 
|  |  | 
|  | static struct platform_driver pic32_dmt_driver = { | 
|  | .probe		= pic32_dmt_probe, | 
|  | .remove		= pic32_dmt_remove, | 
|  | .driver		= { | 
|  | .name		= "pic32-dmt", | 
|  | .of_match_table = of_match_ptr(pic32_dmt_of_ids), | 
|  | } | 
|  | }; | 
|  |  | 
|  | module_platform_driver(pic32_dmt_driver); | 
|  |  | 
|  | MODULE_AUTHOR("Purna Chandra Mandal <purna.mandal@microchip.com>"); | 
|  | MODULE_DESCRIPTION("Microchip PIC32 DMT Driver"); | 
|  | MODULE_LICENSE("GPL"); |