详解linux 看门狗驱动编写

看门狗是linux驱动的一个重要环节。某些特殊的设备,有时候需要放在一些环境恶劣的地方,比如电信设备。但是,任何软件都不可能100%没有bug。如何保证软件在遇到严重bug、死机的时候也能正常运行呢,那么看门狗就是有效的一种方法。看门狗一般要求用户定时喂狗,如果一段时间没有喂狗的话,那么系统就会自动重启。今天,我们就来看看这个看门狗驱动怎么编写?

1、代码目录

?

drivers/watchdog

2、阅读目录下的Kconfig,可以找一个s3c模块macro

?

config HAVE_S3C2410_WATCHDOG

  bool

  help

   This will include watchdog timer support for Samsung SoCs. If

   you want to include watchdog support for any machine, kindly

   select this in the respective mach-XXXX/Kconfig file.

 

config S3C2410_WATCHDOG

  tristate "S3C2410 Watchdog"

  depends on HAVE_S3C2410_WATCHDOG || COMPILE_TEST

  select WATCHDOG_CORE

  select MFD_SYSCON if ARCH_EXYNOS

  help

   Watchdog timer block in the Samsung SoCs. This will reboot

   the system when the timer expires with the watchdog enabled.

 

   The driver is limited by the speed of the system's PCLK

   signal, so with reasonably fast systems (PCLK around 50-66MHz)

   then watchdog intervals of over approximately 20seconds are

   unavailable.

 

   The driver can be built as a module by choosing M, and will

   be called s3c2410_wdt

3、S3C2410_WATCHDOG主要依赖WATCHDOG_CORE,可以继续跟踪Makefile

?

obj-$(CONFIG_S3C2410_WATCHDOG) += s3c2410_wdt.o

4、macro只依赖一个s3c2410_wdt.c文件,继续查看

?

static SIMPLE_DEV_PM_OPS(s3c2410wdt_pm_ops, s3c2410wdt_suspend,

      s3c2410wdt_resume);

 

static struct platform_driver s3c2410wdt_driver = {

  .probe   = s3c2410wdt_probe,

  .remove   = s3c2410wdt_remove,

  .shutdown  = s3c2410wdt_shutdown,

  .id_table  = s3c2410_wdt_ids,

  .driver   = {

    .name  = "s3c2410-wdt",

    .pm = &s3c2410wdt_pm_ops,

    .of_match_table = of_match_ptr(s3c2410_wdt_match),

  },

};

 

module_platform_driver(s3c2410wdt_driver);

5、确认driver为platform类型,继续在probe函数中查找有用的code

?

ret = watchdog_register_device(&wdt->wdt_device);

if (ret) {

  dev_err(dev, "cannot register watchdog (%d)\n", ret);

  goto err_cpufreq;

}

6、网上继续查找,寻找到和watchdog有关的数据结构

?

static const struct watchdog_info s3c2410_wdt_ident = {

  .options     =   OPTIONS,

  .firmware_version = 0,

  .identity     = "S3C2410 Watchdog",

};

 

static const struct watchdog_ops s3c2410wdt_ops = {

  .owner = THIS_MODULE,

  .start = s3c2410wdt_start,

  .stop = s3c2410wdt_stop,

  .ping = s3c2410wdt_keepalive,

  .set_timeout = s3c2410wdt_set_heartbeat,

  .restart = s3c2410wdt_restart,

};

 

static const struct watchdog_device s3c2410_wdd = {

  .info = &s3c2410_wdt_ident,

  .ops = &s3c2410wdt_ops,

  .timeout = S3C2410_WATCHDOG_DEFAULT_TIME,

};

7、找到设备注册函数、函数结构基本就算结束了,当然有中断的话,也可以确认一下

?

ret = devm_request_irq(dev, wdt_irq->start, s3c2410wdt_irq, 0,

      pdev->name, pdev);

if (ret != 0) {

  dev_err(dev, "failed to install irq (%d)\n", ret);

  goto err_cpufreq;

}

8、有兴趣的话,可以找一个函数阅读一下。比如下面这个重启函数,可以和spec对比者来看

?

static int s3c2410wdt_restart(struct watchdog_device *wdd, unsigned long action,

         void *data)

{

  struct s3c2410_wdt *wdt = watchdog_get_drvdata(wdd);

  void __iomem *wdt_base = wdt->reg_base;

 

  /* disable watchdog, to be safe */

  writel(0, wdt_base + S3C2410_WTCON);

 

  /* put initial values into count and data */

  writel(0x80, wdt_base + S3C2410_WTCNT);

  writel(0x80, wdt_base + S3C2410_WTDAT);

 

  /* set the watchdog to go and reset... */

  writel(S3C2410_WTCON_ENABLE | S3C2410_WTCON_DIV16 |

    S3C2410_WTCON_RSTEN | S3C2410_WTCON_PRESCALE(0x20),

    wdt_base + S3C2410_WTCON);

 

  /* wait for reset to assert... */

  mdelay(500);

 

  return 0;

}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。

原文链接:https://blog.csdn.net/feixiaoxing/article/details/79874642

原创文章,作者:JKPOR,如若转载,请注明出处:http://www.wangzhanshi.com/n/6786.html

(0)
JKPOR的头像JKPOR
上一篇 2025年1月1日 16:24:13
下一篇 2025年1月1日 16:24:15

相关推荐

  • Linux开机启动过程详解

    计算机开机是一个神秘的过程。我们只是按了开机键,就看到屏幕上的进度条或者一行行的输出,直到我们到达登录界面。然而,计算机开机又是个异常脆弱的过程,我们满心期望的登录界面可能并不会出…

    Linux 2025年1月1日
  • RHEL 7中防火墙的配置和使用方法

    RHEL7 中使用了firewalld代替了原来的iptables,操作设置和原来有点不同: 查看防火墙状态:systemctl status firewalld 启动防火墙:sy…

    Linux 2025年1月1日
  • 详解在Linux下搭建Git服务器

    众所周知,版本系统在开发环境中是必不可少的,但是我们可以把代码免费的托管到GitHub上,如果我们不原意公开项目的源代码,公司又不想付费使用,那么我们可以自己搭建一台Git服务器,…

    Linux 2025年1月1日
  • Linux与windows文件传输详解及实例

    Linux系统之间传输文件有很多种方法,此篇博客介绍其中的两种。也是在开发过程中经常用到的。 一般情况下,个人经常用到rz或sz命令来上传下载文件。 rz sz 上传下载 安装rz…

    Linux 2025年1月1日
  • Linux进程地址空间详解

    一、C语言内存管理基础 引入:以前我们知道一个指针指向的如果是一个常量字符串,那么这个就是指向的常量区,只读不可被修改,因此下面的程序会崩溃。 1、在我们C语言内存管理机制里面线性…

    2024年12月17日
  • Linux多线程环境下 关于进程线程终止函数总结

    pthread_kill: pthread_kill与kill有区别,是向线程发送signal。,大部分signal的默认动作是终止进程的运行,所以,我们才要用signal()去抓…

    Linux 2025年1月1日
  • Linux使用 diff 和 patch 命令协同开发

    大家好,我是良许。 之前我在公司上班的时候,需要经常跟外国同事一起协同开发(之前在外企上班)。由于是异地协作,所以沟通几乎全部是通过邮件。 我们有使用 Git 进行代码版本管理,但…

    2025年1月1日
  • linux虚拟网络设备之vlan配置详解

    简介 VLAN是网络栈的一个附加功能,且位于下两层。首先来学习Linux中网络栈下两层的实现,再去看如何把VLAN这个功能附加上去。下两层涉及到具体的硬件设备,日趋完善的Linux…

    2025年1月1日
  • 关于g++和gcc的相同点和区别详解

    gcc和g++的区别和联系 gcc和g++都是GNU(一个组织)的编译器。 1、对于.c后缀的文件,gcc把它当做是C程序;g++当做是C++程序; 2、对于.cpp后缀的文件,g…

    Linux 2025年1月1日
  • Linux中环境变量配置的步骤详解

    简介 我们大家在平时使用linux的时候,经常需要配置一些环境变量,这时候一般都是网上随便搜搜就有人介绍经验的。不过问题在于他们的方法各不相同,有人说配置在/etc/profile…

    2025年1月1日

发表回复

登录后才能评论