Linux驱动开发笔记(九)——内核定时器

发布于:2025-08-31 ⋅ 阅读:(18) ⋅ 点赞:(0)

视频:第12.1讲 Linux内核定时器实验-内核时间管理简介_哔哩哔哩_bilibili

文档:《【正点原子】I.MX6U嵌入式Linux驱动开发指南V1.81.pdf》五十章


 1. 内核时间管理

《指南》50.1.1节:

        Linux内核中有大量的函数需要时间管理,比如周期性的调度程序、延时程序、定时器。硬件定时器提供时钟源,时钟源的频率可以设置, 设置好以后周期性的产生定时中断,系统使用定时中断来计时。中断周期性产生的频率就是系统频率, 也叫做节拍率(tick rate)(有的资料也叫系统频率),比如1000Hz,100Hz等等说的就是系统节拍率。

1.1 设置系统节拍率

        编译Linux内核的时候可以通过图形化界面设置系统节拍率:

cd /.../linux/linux-imx-rel_imx_4.1.15_2.1.0_ga_alientek
make menuconfig

        选择kernel features -> Timer frequency,然后便可选择系统街拍率:

        使用命令vi .config去查看:

1.2 系统时间计数器 jiffies

        系统会有一个全局变量jiffies记录启动以来的系统节拍数,在系统启动时初始化为0

        为了兼容32位和64位系统,定义了32位和64位两种jiffies,32位jiffies其实就是64位jiffies的低32位:

// 定义在include/linux/jiffies.h
extern u64 __jiffy_data jiffies_64; 
extern unsigned long volatile __jiffy_data jiffies; 
get_jiffies_64() 读取jiffies的值。32位和64位系统都可使用

        如果jiffies达到最大值,就会从零开始计数,称为溢出。若最小HZ=100,则32位的jiffies约497天溢出;最大HZ=1000,则32位的jiffies约49.7天溢出。64位的jiffies有生之年不会溢出。因此对于32位的jiffies,有一个处理溢出的问题。linux提供了相关API函数:

函数名 返回值 功能
time_after(a, b) (long)(b) - (long)(a) < 0 判断a是否在b之后(a > b ?)
time_before(a, b) time_after(b, a) 判断a是否在b之前(a < b ?)
time_after_eq(a, b) (long)(a) - (long)(b) >= 0) 判断a是否在b之后或等于b(a ≥ b ?)
time_before_eq(a, b) time_after_eq(b, a) 判断a是否在b之前或等于b(a ≤ b ?)
一般都将jiffies填到a,要比较的值填到b

为了方便开发,Linux内核提供了几个jiffies和ms、us、ns之间的转换函数:

示例:

        《指南》中的示例,判断是否超时2s以上:

        等待1ms:

	timeout = jiffies + msecs_to_jiffies(1);
	while (time_before(jiffies, timeout)) {
        …………
    }

2. 内核定时器

《指南》50.1.2节:

        Linux内核定时器采用系统时钟来实现,并不是硬件定时器。Linux内核定时器使用很简单,只需要提供超时时间(相当于定时值)和定时处理函数即可,当超时时间到了以后设置的定时处理函数就会执行,不需要做一大堆的寄存器初始化工作。

        在使用内核定时器的时候要注意一点,内核定时器并不是周期性运行的,超时以后就会自动关闭,因此如果想要实现周期性定时,那么就需要在定时处理函数中重新开启定时器。

timer_list结构体表示内核定时器,定义如下:

// 定义在include/linux/timer.h
struct timer_list {
	struct list_head entry;
	unsigned long expires;   // 超时时间,单位是节拍数
	struct tvec_base *base;

	void (*function)(unsigned long);    // 超时处理函数
	unsigned long data;     // 要传递给function函数的参数

	int slack;
};

2.1 内核定时器API函数

2.1.1 init_timer

        初始化定时器,但不会启动。

        这个函数并不能配置定时器结构体里的entry、expires、data等变量,只是将这个timer_list结构体清空。

void init_timer(struct timer_list *timer) 
// timer: 要初始化的定时器

2.1.2 add_timer

        向Linux内核注册/启动定时器。注册以后定时器就会开始运行。

        必须先init_timer初始化,并配置好定时器结构体里的参数,再注册。

void add_timer(struct timer_list *timer)
//timer:要注册的定时器

2.1.3 del_timer

        删除一个定时器(不管定时器有没有被激活都可以删除。但是在多处理器系统上,定时器可能会在其他的处理器上运行,因此在调用del_timer之前要先等待其他处理器的定时处理器函数退出)。

int del_timer(struct timer_list * timer)
// timer:  要删除的定时器
// return: 0,定时器还没激活;1,定时器已激活

2.1.4 del_timer_sync

        也是删除定时器,但是会等待其他处理器使用完该定时器再删除。

         del_timer_sync不能在中断服务函数使用。

int del_timer_sync(struct timer_list *timer)
// timer:  要删除的定时器
// return: 0,定时器还没激活;1,定时器已激活

2.1.5 mod_timer

        修改定时值。

        如果定时器还没有激活,那么mod_timer函数会激活定时器。

int mod_timer(struct timer_list *timer, unsigned long expires)
//timer:  要修改的定时器。 
//expires:修改后的超时时间 
//return: 0,调用mod_timer函数前定时器未被激活;1,调用mod_timer函数前定时器已被激活。 

2.1.6 示例

struct timer_list timer; /* 定义定时器 */

/* 定时器回调函数 */
void function(unsigned long arg){
    /*
     * 定时器处理代码
     */

    /* 如果需要定时器周期性运行的话就使用mod_timer重新设置超时值并且启动定时器 */
    mod_timer (&dev->timertest, jiffies + msecs_to_jiffies (2000));
}

/* 初始化函数 */
void init(void){
    init_timer(&timer); /* 初始化定时器 */
    timer.function = function; /* 设置定时处理函数 */
    timer.expires=jffies + msecs_to_jiffies(2000);/* 超时时间2秒 */
    timer.data = (unsigned long)&dev; /* 将设备结构体作为参数 */

    add_timer (&timer); /* 注册 /启动定时器 */
}

/* 退出函数 */
void exit(void){
    del_timer(&timer); /* 删除定时器 */
    /* 或者使用del_timer_sync(&timer); */
}

2.2 短延时函数

3. 定时器实验

3.1 文件结构

13_TIMER (工作区)
├── .vscode
│   ├── c_cpp_properties.json
│   └── settings.json
├── 13_timer.code-workspace
├── Makefile
└── timer.c

3.2 Makefile

CFLAGS_MODULE += -w

KERNELDIR := /home/for/linux/imx6ull/linux/linux-imx-rel_imx_4.1.15_2.1.0_ga_alientek  # 内核路径
# KERNELDIR改成自己的 linux-imx-rel_imx_4.1.15_2.1.0_ga_alientek文件路径(这个文件从正点原子“01、例程源码”中直接搜,cp到虚拟机里面)
CURRENT_PATH := $(shell pwd)	# 当前路径
obj-m := timer.o			# 编译文件
build: kernel_modules			# 编译模块
kernel_modules:
	$(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) modules
clean:
	$(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) clean	

3.3 timer.c

主要有以下更新:

        增加了 定时器处理函数timer_func()

        增加了 设备结构体中 定时器定义

        增加了 驱动入口函数中 定时器初始化

        增加了 驱动出口函数中 定时器删除

        增加了 LED初始化函数led_init()

gpio和pinctrl配置与Linux驱动开发笔记(六)中的实验配置一致,这里不再重复(记得设备树注释掉重复的io)

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/fs.h>
#include <linux/uaccess.h>
#include <linux/io.h>
#include <linux/cdev.h>
#include <linux/kdev_t.h>
#include <linux/stat.h>
#include <linux/device.h>
#include <linux/of_gpio.h>
#include <linux/atomic.h>
#include <linux/timer.h>
#include <linux/jiffies.h>


#define DEV_CNT  1
#define DEV_NAME "timer"


/* 设备结构体 */
struct timer_dev{
    dev_t devid;
    int major;
    int minor;
    struct cdev cdev;
    struct device *device;
    struct class *class;
    struct device_node *nd;
    struct timer_list timer;
    int led_gpio;
};
struct timer_dev timerdev;

/* 定时器处理函数 */
static void timer_func(unsigned long arg){
    struct timer_dev *dev = (struct timer_dev*)arg;
    static int state = 1;
    state = !state;
    gpio_set_value(dev->led_gpio, state);

    mod_timer(&dev->timer, jiffies + msecs_to_jiffies(500));
}
/* 初始化LED */
int led_init(struct timer_dev *dev){
    int ret = 0;

    /* 1.获取设备节点 */
    dev->nd = of_find_node_by_path("/gpioled");  // 找到刚才在imx6ull-alientek-emmc.dts根节点下加入的gpioled节点
     if(dev->nd == NULL){
        ret = -EINVAL;
        goto fail_fd;
    }

    /* 2.获取LED对应的GPIO */  // 也就是节点中led-gpios那一行内容
    dev->led_gpio = of_get_named_gpio(dev->nd, "led-gpios", 0);
    if(dev->led_gpio < 0){
        ret = -EINVAL;
        goto fail_gpio;
    }
 
    /* 3.申请IO */
    ret = gpio_request(dev->led_gpio, "led-gpios");
    if(ret){
        ret = -EBUSY;
        printk("IO %d busy!\r\n",dev->led_gpio);
        goto fail_request;
    }
 
    /* 4.使用IO,设置为输出 */
    ret = gpio_direction_output(dev->led_gpio, 1);
    if(ret < 0){
        ret = -EINVAL;
        goto fail_gpioset;
    }
    return 0;

fail_gpioset:
    gpio_free(dev->led_gpio);
fail_request:
fail_gpio:
fail_fd:
    return ret;
}


/* 操作集 */
static int key_release(struct inode *inode, struct file *filp){
    struct key_dev *dev = filp->private_data;

    return 0;
}
static int key_open(struct inode *inode, struct file *filp){
    filp->private_data = &timerdev;

    return 0;
}
static ssize_t key_write(struct file *filp, const char __user *buf, size_t count, loff_t *ppos){
    int ret;

    return 0;
}
static ssize_t key_read(struct file *filp, char __user *buf, size_t count, loff_t *ppos){
    struct key_dev *dev = filp->private_data;
    int ret = 0; 
    return 0;
}
static const struct file_operations key_fops = {
    .owner = THIS_MODULE,
    .write = key_write,
    .open = key_open,
    .release = key_release,
    .read = key_read,
};


/* 驱动入口 */
static int __init key_init(void){
    int ret = 0;

    /* 1.注册字符设备驱动 */
    timerdev.devid = 0;
    if(timerdev.devid){
        timerdev.devid = MKDEV(timerdev.devid, 0);
        register_chrdev_region(timerdev.devid, DEV_CNT, DEV_NAME);
    } else {
        alloc_chrdev_region(&timerdev.devid, 0, DEV_CNT, DEV_NAME);
        timerdev.major = MAJOR(timerdev.devid);
        timerdev.minor = MINOR(timerdev.devid);
    }

    /* 2.初始化cdev */
    timerdev.cdev.owner = THIS_MODULE;  
    cdev_init(&timerdev.cdev, &key_fops);

    /* 3.添加cdev */
    cdev_add(&timerdev.cdev, timerdev.devid, DEV_CNT); // 错误处理先略过了

    /* 4.创建类 */
    timerdev.class = class_create(THIS_MODULE, DEV_NAME);
    if(IS_ERR(timerdev.class)){
        return PTR_ERR(timerdev.class);
    }

    /* 5.创建设备 */
    timerdev.device = device_create(timerdev.class, NULL, timerdev.devid, NULL, DEV_NAME);
    if(IS_ERR(timerdev.device)){
        return PTR_ERR(timerdev.device);
    }

    /* 6. 初始化LED */
    ret = led_init(&timerdev);
    if(ret < 0){
        ret = -EINVAL;
        goto fail_ledinit;
    }

    /* 7. 初始化定时器 */
    init_timer(&timerdev.timer);
    timerdev.timer.function = timer_func;
    timerdev.timer.expires = jiffies + msecs_to_jiffies(500); // 延迟500ms
    timerdev.timer.data = (unsigned long)&timerdev;
    add_timer(&timerdev.timer);

    return 0;

fail_ledinit:
fail_device:
    class_destroy(timerdev.class);
fail_class:
    cdev_del(&timerdev.cdev);

    return ret;
}

/* 驱动出口 */
static void __exit key_exit(void){
    /* 关灯 */
    gpio_set_value(timerdev.led_gpio, 1);

    /* 删除定时器*/
    del_timer(&timerdev.timer);

    /* 注销字符设备驱动 */
    cdev_del(&timerdev.cdev);
    unregister_chrdev_region(timerdev.devid, DEV_CNT);

    device_destroy(timerdev.class, timerdev.devid);
    class_destroy(timerdev.class);

    /* 释放io */
    gpio_free(timerdev.led_gpio);
}

module_init(key_init);
module_exit(key_exit);
MODULE_LICENSE("GPL");

3.4 测试

# VSCODE
make
sudo cp timer.ko /home/for/linux/nfs/rootfs/lib/modules/4.1.15/ -f

# 串口
cd /lib/modules/4.1.15/
depmod
modprobe timer.ko  # 此时开发板红灯开始闪烁

网站公告

今日签到

点亮在社区的每一天
去签到