内核定时器timer_list

内核在时钟中断发生后执行检测各个定时器是否到期,到期后的定时器处理函数将作为软中断在底半部执行。实质上,时钟中断处理程序会唤起TIMER_SOFTIRQ软中断,运行当前处理器上到期的所有定时器。linux提供的内核定时器数据结构为timer_list。

一. 定义

timer_list定义在linux/timer.h中,实现在kernel/timer.c中。

struct timer_list {
    struct list head_entry;
    unsigned long expires;    //到期时间
    struct tves_base *base;

    void (*func)(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

};

内核相关定义:HZ和jiffies

HZ  时钟频率
jiffies 32位无符号数,当前时钟计数

# define HZ     CONFIG_HZ   /* Internal kernel timer frequency */
# define USER_HZ    100     /* some user interfaces are */
linux/jiffies.h
/*
 * The 64-bit value is not atomic - you MUST NOT read it
 * without sampling the sequence number in jiffies_lock.
 * get_jiffies_64() will do this for you as appropriate.
 */
extern u64 __jiffy_data jiffies_64;
extern unsigned long volatile __jiffy_data jiffies;

#if (BITS_PER_LONG < 64) 
u64 get_jiffies_64(void);
#else
static inline u64 get_jiffies_64(void)
{
    return (u64)jiffies;
}
#endif

二. 函数

初始化定时器

init_timer(&mytimer);

DEFINE_TIMER(_name, _function, _expires, _data);

TIMER_INITIALIZER(_function, _expires, _data);

setup_timer(timer, fn, data);

首次使能定时器

add_timer(&mytimer);
再次使能定时器

mod_timer(&mytimer, unsigned long expires);
删除定时器

del_timer_sync(&mytimer); //删除前要等待定时器处理完,不能发生在中断上下文。

del_timer(&mytimer); //删除定时器

三. 示例

#include <linux/init.h>
#include <linux/module.h>
#include <asm/current.h>
#include <linux/timer.h>
#inlcude <linux/sched.h>

MODULE_LISENCE("GPL");

extern struct task_struct *current;

struct timer_list mytimer;

void timer_handler(unsigned long arg)
{
    printk("timer arg = %d\n", arg);
    mod_timer(&mytimer, jiffies + HZ * 5);

    return;
}

static int __init akae_init(void)
{
    int local = 0;

    printk("module name is %s\n", KBUILD_MODNAME);
    printk("akae_init at %p\n", akae_init);

    printk("jiffies at %p\n", &jiffies);
    printk("jiffies is %ld\n", jiffies);

    printk("The process is \"%s\" (pid %i)\n", current->name, curent->pid);
    
    init_timer(&mytimer);

    printk("HZ is %d\n", HZ);
    mytimer.expires = jiffies + HZ * 5;
    mytimer.function = timer_handler;
    mytimer.data = 100;

    add_timer(&mytimer);

    return 0;
}

static void __exit akae_exit(void)
{
    printk("module eixt.\n");
    printk("akae_exit at %p\n", akae_exit);

    del_timer_sync(&mytimer);

    return;
}

module_init(akae_init);
module_exit(akae_exit);
Linux 定时器 timer_list 是一个内核数据结构,用于管理内核中的定时器。它是一个双向链表,每个节点表示一个定时器timer_list 的定义位于 `<linux/timer.h>` 头文件中。 每个 timer_list 节点的定义如下: ```c struct timer_list { struct list_head entry; // 定时器节点的链表指针 unsigned long expires; // 定时器的到期时间 void (*function)(unsigned long); // 定时器回调函数 unsigned long data; // 传递给回调函数的参数 struct tvec_base *base; // 定时器所属的时间轮 int slack; // 定时器的松弛时间 }; ``` 其中,`entry` 是一个 `list_head` 结构,用于将节点连接到定时器链表中。`expires` 表示定时器的到期时间,以 jiffies 单位表示。`function` 是定时器的回调函数,在定时器到期时被调用。`data` 是传递给回调函数的参数。`base` 表示定时器所属的时间轮,`slack` 是定时器的松弛时间,用于处理定时器的精度。 在使用 timer_list 时,可以使用以下函数进行初始化和操作: - `timer_setup(struct timer_list *timer, void (*function)(unsigned long), unsigned int flags)`:初始化一个定时器,并指定回调函数和标志。 - `init_timer(struct timer_list *timer)`:初始化一个定时器。 - `add_timer(struct timer_list *timer)`:将定时器添加到定时器链表中。 - `del_timer(struct timer_list *timer)`:从定时器链表中删除定时器。 - `mod_timer(struct timer_list *timer, unsigned long expires)`:修改定时器的到期时间。 这些函数可以通过 `<linux/timer.h>` 头文件中的宏来调用。通过操作 timer_list,可以实现在 Linux 内核中的定时器功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值