手把手教你写Linux设备驱动---定时器(一)(基于友善之臂4412开发板)

简介: 手把手教你写Linux设备驱动---定时器(一)(基于友善之臂4412开发板)

这个专题我们来说下Linux中的定时器。


Linux内核中,有这样的一个定时器,叫做内核定时器,内核定时器用于控制某个函数,也就是定时器将要处理的函数在未来的某个特定的时间内执行。内核定时器注册的处理函数只执行一次,即不是循环执行的。

如果对延迟的精度要求不高的话,最简单的实现方法如下---忙等待:

Unsigned long  j = jiffies + jit_delay * HZ;
While(jiffies  <  j)
{
         ……
}

下面来说下具体的参数代表的含义:

jiffies:全局变量,用来记录自系统启动以来产生的节拍总数。启动时内核将该变量初始化为0

此后每次时钟中断处理程序增加该变量的值。每一秒钟中断次数HZjiffies一秒内增加HZ。系统运行时间 = jiffie/HZ.

jiffies用途:计算流逝时间和时间管理

jiffies内部表示:

extern u64 jiffies_64;
extern unsigned long volatilejiffies;     //位长更系统有关32/64---->
|
|

32位:497天后溢出

64位:……

在定时器中有这样一个概念,度量时间差:

时钟中断由系统的定时硬件以周期性的时间间隔产生,这个间隔说白了其实就是频率由内核根据HZ来确定,HZ是一个与体系结构无关的常数,可以配置为(50-1200),X86平台,它的值被默认为1000 ;

定时器在内核中相关的头文件以及数据结构如下:

#include <linux/timer.h>  /*timer*/
#include <asm/uaccess.h>  /*jiffies*/
struct timer_list {
  /*
   * All fields that change during normal runtime grouped to the
   * same cacheline
   */
   //定时器可以作为链表的一个节点
  struct list_head entry;
  //定时值基于jiffies
  unsigned long expires;
  //定时器内部值
  struct tvec_base *base;
  //定时器处理函数
  void (*function)(unsigned long);
   //定时器处理函数参数
  unsigned long data;
  int slack;
#ifdef CONFIG_TIMER_STATS
  int start_pid;
  void *start_site;
  char start_comm[16];
#endif
#ifdef CONFIG_LOCKDEP
  struct lockdep_map lockdep_map;
#endif
};

定时器最基本的使用方法可以使用下面这两个个内核提供的宏:

//初始化定时器
#define init_timer(timer)\
  init_timer_key((timer), NULL, NULL)
//注册一个定时器
#define setup_timer(timer, fn, data)\
  setup_timer_key((timer), NULL, NULL, (fn), (data))

还有以下两个函数:


添加一个定时器

void add_timer(struct timer_list *timer)

删除一个定时器

int del_timer(struct timer_list *timer)

那么写一个定时器的具体步骤是什么?
1、初始化内核定时器
2、设置定时器执行函数的参数(可有可无)
3、设置定时时间
4、设置定时器函数
5、启动定时器


接下来,我们结合一个简单的驱动来了解这个过程,这个驱动非常简单,就是开机后,5s钟后,开发板上的蜂鸣器就会每隔1s钟交替响。

先来看看开发板的蜂鸣器的原理图:


(1)蜂鸣器接口位于电路板的底板,看电路图可知道是高电平有效。

640.png

(2)相对应的找到核心板的接口。由此可知,我们的蜂鸣器是GPD0_0

640.jpg

接下来找数据手册,找到对应的寄存器,然后配置它就可以了。

 2、查数据手册,找到相关的寄存器,并配置0.gif

(1)找到GPD0CON,地址是0x114000A0,我们需要配置GPD0CON(0)为输出状态。也就是写0x1这个值到这个寄存器。

640.jpg

(2)找到GPD0DAT这个寄存器,用于配置蜂鸣器的高低电平,物理地址是0x114000A4,刚好与上一个差4个字节的偏移

我们只要对这个寄存器写1和写0,那么蜂鸣器就可以叫起来了,哈哈。是不是很简单?0.gif

640.jpg

整个简单的驱动代码如下:

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/platform_device.h>
#include <linux/fb.h>
#include <linux/backlight.h>
#include <linux/err.h>
#include <linux/pwm.h>
#include <linux/slab.h>
#include <linux/miscdevice.h>
#include <linux/delay.h>
#include <linux/gpio.h>
#include <mach/gpio.h>
#include <plat/gpio-cfg.h>
#include <linux/timer.h>  /*timer*/
#include <asm/uaccess.h>  /*jiffies*/
#include <linux/delay.h>
//设备名称
#define DEVICE_NAME       "Bell"
//设备GPIO引脚
#define BUZZER_GPIO     EXYNOS4_GPD0(0)
//定义一个定时器链表
struct timer_list timer;
static void Bell_init()
{
  //1、请求gpio,相当于注册gpio
  gpio_request(BUZZER_GPIO,DEVICE_NAME);
  //2、调用板级驱动的函数,将gpio配置成输出状态
  s3c_gpio_cfgpin(BUZZER_GPIO, S3C_GPIO_OUTPUT);
        //3、设置gpio为0,表示低电平,蜂鸣器高电平就会响
  gpio_set_value(BUZZER_GPIO,0);
}
void timer_function(unsigned long value)
{
  while(value)  
  {
    //设置gpio为1,表示高电平,蜂鸣器高电平就会响
    gpio_set_value(BUZZER_GPIO,1);
    printk("BUZZER ON\n");
    mdelay(1000);
    //设置gpio为0,表示低电平,蜂鸣器高电平就会响
    gpio_set_value(BUZZER_GPIO,0);
    printk("BUZZER OFF\n");
    mdelay(1000);
  }
}
static int __init tiny4412_Bell_init(void) 
{
    //bell init
    Bell_init();  
    //初始化内核定时器
    init_timer(&timer); 
    //给执行的函数传参  
    timer.data= 1;
    //当前jiffies的值加上5秒钟之后  
    timer.expires= jiffies + (5 * HZ);
    //如果超时了就执行这个函数
    timer.function= timer_function;
    //启动定时器
    add_timer(&timer);                        
    return 0 ;
}
static void __exit tiny4412_Bell_exit(void) 
{
    //释放gpio
    gpio_free(BUZZER_GPIO);
    //删除注册的定时器
    del_timer(&timer);
}
module_init(tiny4412_Bell_init);
module_exit(tiny4412_Bell_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("YYX");
MODULE_DESCRIPTION("Exynos4 BELL Driver");

接下来,开启我们开发板串口,观察运行结果:

果然,定时器在开发板启动后的若干时间后,就周而复始的去打开和关闭我们板子上的蜂鸣器了。

640.png

目录
相关文章
|
22天前
|
Linux 网络安全 网络虚拟化
Linux虚拟网络设备:底层原理与性能优化深度解析
在深入探讨Linux虚拟网络设备的底层原理之前,重要的是要理解这些设备如何在Linux内核中实现,以及它们如何与操作系统的其他部分交互以提供高效且灵活的网络功能。虚拟网络设备在现代网络架构中发挥着关键作用🔑,特别是在云计算☁️、容器化📦和网络功能虚拟化(NFV)环境中。
Linux虚拟网络设备:底层原理与性能优化深度解析
|
22天前
|
Linux 网络虚拟化 虚拟化
Linux虚拟网络设备深度解析:使用场景、分类与开发者指南
Linux虚拟网络设备支撑着各种复杂的网络需求和配置,从基础的网络桥接到高级的网络隔离和加密🔐。以下是对主要Linux虚拟网络设备的介绍、它们的作用以及适用场景的概览,同时提出了一种合理的分类,并指出应用开发人员应该着重掌握的设备。
Linux虚拟网络设备深度解析:使用场景、分类与开发者指南
|
23天前
|
安全 Linux API
Linux设备模型统一:桥接硬件多样性与应用程序开发的关键
在Linux的宏大世界中,各种各样的硬件设备如星辰般繁多。从常见的USB设备到复杂的网络接口卡,从嵌入式设备到强大的服务器,Linux需要在这些差异极大的硬件上运行。这就引出了一个问题:Linux是如何统一这些不同硬件的设备模型的呢?本文将探讨Linux是如何针对不同的硬件统一设备模型的,这一统一的设备模型对于应用程序开发人员来说又有何意义。让我们一探究竟🕵️‍♂️。
Linux设备模型统一:桥接硬件多样性与应用程序开发的关键
|
22天前
|
Cloud Native Linux 网络虚拟化
深入理解Linux veth虚拟网络设备:原理、应用与在容器化架构中的重要性
在Linux网络虚拟化领域,虚拟以太网设备(veth)扮演着至关重要的角色🌐。veth是一种特殊类型的网络设备,它在Linux内核中以成对的形式存在,允许两个网络命名空间之间的通信🔗。这篇文章将从多个维度深入分析veth的概念、作用、重要性,以及在容器和云原生环境中的应用📚。
深入理解Linux veth虚拟网络设备:原理、应用与在容器化架构中的重要性
|
23天前
|
监控 中间件 Linux
深入Linux设备模型:开发者指南
Linux的设备模型是操作系统管理硬件设备的一种高级抽象,它不仅涉及到设备驱动程序的加载和卸载,还包括设备之间的关系、设备的状态管理以及与用户空间通信的机制。理解Linux的设备模型对于应用开发人员来说至关重要,它有助于开发出更加稳定、高效的应用程序。🌟
深入Linux设备模型:开发者指南
|
3天前
|
存储 监控 Linux
【专栏】如何在 Linux 中列出已安装的驱动器?
【4月更文挑战第28天】在 Linux 中,了解已安装驱动器是系统管理的关键。本文介绍了三种方法:1) 使用 `lsblk` 命令显示设备名、大小和类型;2) `fdisk -l` 命令提供详细分区信息;3) `gnome-disks` 等系统管理工具展示驱动器信息。此外,还讨论了驱动器类型识别、挂载点概念及其应用。通过这些方法,用户能有效地监控和管理 Linux 系统中的驱动器。
|
10天前
|
存储 Linux
如何查看Linux设备的硬盘信息?
【4月更文挑战第12天】在Linux系统中,查看硬盘信息的常用命令。
23 4
|
16天前
|
Linux Go
Linux命令Top 100驱动人生! 面试必备
探索Linux命令不再迷茫!本文分10部分详解20个基础命令,带你由浅入深掌握文件、目录管理和文本处理。 [1]: <https://cloud.tencent.com/developer/article/2396114> [2]: <https://pan.quark.cn/s/865a0bbd5720> [3]: <https://yv4kfv1n3j.feishu.cn/docx/MRyxdaqz8ow5RjxyL1ucrvOYnnH>
68 0
|
22天前
|
网络协议 Linux SDN
虚拟网络设备与Linux网络协议栈
在现代计算环境中,虚拟网络设备在实现灵活的网络配置和隔离方面发挥了至关重要的作用🔧,特别是在容器化和虚拟化技术广泛应用的今天🌐。而Linux网络协议栈则是操作系统处理网络通信的核心💻,它支持广泛的协议和网络服务🌍,确保数据正确地在网络中传输。本文将深入分析虚拟网络设备与Linux网络协议栈的关联,揭示它们如何共同工作以支持复杂的网络需求。
|
22天前
|
Linux SDN 网络虚拟化
Linux虚拟网络设备全景解析:定义、工作模式与实践应用
在深入探索Linux操作系统的强大功能时,我们不可避免地会遇到虚拟网络设备的概念。这些设备扮演着构建和维护虚拟化环境中网络通信的关键角色。本文旨在详细介绍Linux虚拟网络设备的定义、工作模式以及它们的多样化用途。
Linux虚拟网络设备全景解析:定义、工作模式与实践应用

热门文章

最新文章