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

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

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

---

忙等待:

Unsigned long j = jiffies + jit_delay * HZ;

While(jiffies < j)

{

……

}

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

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

此后每次时钟中断处理程序增加该变量的值。每一秒钟中断次数HZ,jiffies一秒内增加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  /*timer*/

#include  /*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)蜂鸣器接口位于电路板的底板,看电路图可知道是高电平有效。

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

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

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

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

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

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

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

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include /*timer*/

#include /*jiffies*/

#include

//设备名称

#define DEVICE_NAME"Bell"

//设备GPIO引脚

#define BUZZER_GPIOEXYNOS4_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");接下来,开启我们开发板串口,观察运行结果:

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

本文同步分享在 博客“Engineer-Bruce_Yang”(CSDN)。

如有侵权,请联系 support@oschina.cn 删除。

本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。

手把手教linux设备驱动,手把手教你写Linux设备驱动---定时器(一)(基于友善之臂4412开发板)...相关推荐

  1. Linux字符型驱动开发—基于友善之臂2416开发板

    驱动程序(Device Driver)是一种可以使计算机和设备通信的特殊程序,相当于内核和硬件之间的接口,操作系统只能通过这个接口,才能控制硬件设备的工作.驱动程序接受上层软件(应用程序.内核)的请求 ...

  2. 4412开发板学习之Linux驱动开发(八):GPIO读操作与按键轮询实现

    GPIO读操作与按键轮询实现 GPIO读操作 硬件 查找对应IO口 寄存器配置 软件 需要的函数 注册设备 代码及分析 实验效果 按键轮询实现 原理分析 硬件 软件 用到的函数 先前准备工作 代码及分 ...

  3. 迅为4412开发板-实验LEDS驱动一

    14.1 本章导读 本节实验介绍一个完整的 GPIO 驱动,以后在 Linux 中需要处理 GPIO 驱动都可以仿照或 者移植这个驱动. 14.1.1 工具 14.1.1.1 硬件工具 1)iTOP4 ...

  4. 迅为4412开发板-步进电机-驱动和测试例程

    迅为4412开发板-步进电机-驱动和测试例程 和本文档配套的资料有: 1 内核配置和步进电机驱动补丁包: topeet_modules_20190319.tar.gz 2 linux-C 的测试例程: ...

  5. 友善之臂编linux内核,手把手教你从头开始搭建友善之臂ARM-tiny4412开发环境(史上最详细!!)...

    在命令行中输入: minicom -s ┌─────[configuration]──────┐ │ Filenames and paths │ File transferprotocols │ Se ...

  6. 手把手教你从头开始搭建友善之臂ARM-tiny4412开发环境(史上最详细!!)

    创建一个ARM目录 mkdir   /disk/A9  -p 接下来你需要准备以下的东西 1.arm-linux-gcc-4.5.1     交叉编译器 2.linux-3.5-tiny4412    ...

  7. 讯为4412开发板Linux驱动学习笔记

    驱动理论专题一 Linux驱动程序的基本认识 有了内存管理单元,就有虚拟地址,物理地址. 驱动理论专题二 学会查看原理图 以LED2为示例 通过原理图查看到KP_COL0,赋予高电平则能点亮LED2, ...

  8. 4412开发板学习之Linux驱动开发(五):4412MMU及GPIO操作(点灯)

    4412MMU及GPIO操作(点灯) 物理地址与虚拟地址 与传统MCU的对比 4412中的物理地址 MMU内存管理单元 存储器分类 4412中的存储器映射 物理地址和虚拟地址 其他的地址概念 GPIO ...

  9. linux 4412 adc字符设备,迅为4412开发板Linux字符设备控制(二)

    17.3  字符类 Buzzer  蜂鸣器 和 led 灯类似,蜂鸣器的设备节点也是在/dev 目录下,如下图所示. 蜂鸣器的硬件和 led 灯类似,如下图所示. 如上图所示. 原理图很容易理解,如果 ...

最新文章

  1. Redis源码解析——双向链表
  2. 从课堂走向实践还有多远?
  3. 2007年分区联赛提高组之一 统计数字
  4. python学到什么程度算是会-Python 必须学到什么程度?
  5. linux web 服务器性能,Linux系统Web服务器性能测试(2)
  6. html 基础之canvas 和 localStorage
  7. Java 压缩解压字符串(支持中文)
  8. 【LeetCode笔记】399. 除法求值(Java、图)
  9. 创业人永远不要让工作成为自己的负担
  10. 甘特图怎么做项目进度计划
  11. 匿名四轴上位机使用方法
  12. 针式打印机保养方法汇总
  13. 解决Substrate节点模板编译失败ailed to run custom build command for `tikv-jemalloc-sys v0.4.3+5.2.1-patched.2`
  14. 读书笔记——好句摘抄
  15. 1.DingoApi安装和配置
  16. 10 分钟搞定 Vim 编辑器常用命令!
  17. 卢松松博客专访胡茬:解密程序员互联网创业历程
  18. 记录自己在pyqt5用电脑摄像头拍照并保存照片过程中发现的问题
  19. Sass学习笔记 — 4. 参考手册
  20. 使用开源软件Prometheus监控企业内部应用

热门文章

  1. 从零开始前端学习[38]:html5中的弹性布局一(移动端响应式实现各种布局,极其重要)
  2. RIP路由协议中的split-horizon与split-horizon with poisoned reverse剖析
  3. pip install mpi4py报错:ERROR: Could not build wheels for mpi4py, which is required to install pyprojec
  4. 什么是TLB ?(转载)
  5. MFS分布式文件系统一
  6. 用Lua实现基于观察者模型的游戏成就系统
  7. NBS的N把“瑞士军刀”——产品解读之规则设计器
  8. 力天创见客流分析功能需求
  9. 打开损坏的Word文档-word修复_目前只用过打开并修复
  10. h5应用 vue 钉钉_钉钉企业内部H5微应用开发