cregit-Linux how code gets into the kernel

Release 4.14 drivers/watchdog/pic32-dmt.c

Directory: drivers/watchdog
/*
 * 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)); }

Contributors

PersonTokensPropCommitsCommitProp
Purna Chandra Mandal26100.00%1100.00%
Total26100.00%1100.00%


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(); }

Contributors

PersonTokensPropCommitsCommitProp
Purna Chandra Mandal30100.00%1100.00%
Total30100.00%1100.00%


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; }

Contributors

PersonTokensPropCommitsCommitProp
Purna Chandra Mandal47100.00%1100.00%
Total47100.00%1100.00%


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); }

Contributors

PersonTokensPropCommitsCommitProp
Purna Chandra Mandal81100.00%1100.00%
Total81100.00%1100.00%


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; }

Contributors

PersonTokensPropCommitsCommitProp
Purna Chandra Mandal44100.00%1100.00%
Total44100.00%1100.00%


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; }

Contributors

PersonTokensPropCommitsCommitProp
Purna Chandra Mandal64100.00%1100.00%
Total64100.00%1100.00%


static int pic32_dmt_start(struct watchdog_device *wdd) { struct pic32_dmt *dmt = watchdog_get_drvdata(wdd); dmt_enable(dmt); return dmt_keepalive(dmt); }

Contributors

PersonTokensPropCommitsCommitProp
Purna Chandra Mandal32100.00%1100.00%
Total32100.00%1100.00%


static int pic32_dmt_stop(struct watchdog_device *wdd) { struct pic32_dmt *dmt = watchdog_get_drvdata(wdd); dmt_disable(dmt); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Purna Chandra Mandal29100.00%1100.00%
Total29100.00%1100.00%


static int pic32_dmt_ping(struct watchdog_device *wdd) { struct pic32_dmt *dmt = watchdog_get_drvdata(wdd); return dmt_keepalive(dmt); }

Contributors

PersonTokensPropCommitsCommitProp
Purna Chandra Mandal27100.00%1100.00%
Total27100.00%1100.00%

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; }

Contributors

PersonTokensPropCommitsCommitProp
Purna Chandra Mandal28498.95%150.00%
Wei Yongjun31.05%150.00%
Total287100.00%2100.00%


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; }

Contributors

PersonTokensPropCommitsCommitProp
Purna Chandra Mandal46100.00%1100.00%
Total46100.00%1100.00%

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

Overall Contributors

PersonTokensPropCommitsCommitProp
Purna Chandra Mandal97499.69%150.00%
Wei Yongjun30.31%150.00%
Total977100.00%2100.00%
Directory: drivers/watchdog
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.