| /* | 
 |  * 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"); |