8

我想在 Linux 设备驱动程序中实现一个计数器,该计数器在每个固定时间间隔后递增。我想在计时器的帮助下做到这一点。示例代码片段将非常有用。

4

3 回答 3

12

查看以下文章IBM Developerworks:计时器和列表

有一个关于如何使用 Linux 内核定时器的小例子(为了方便起见,这里包含了它,评论来自我自己,删除了 printk 消息)

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/timer.h>

MODULE_LICENSE("GPL");

static struct timer_list my_timer;

void my_timer_callback( unsigned long data )
{
     /* do your timer stuff here */
}

int init_module(void)
{
  /* setup your timer to call my_timer_callback */
  setup_timer(&my_timer, my_timer_callback, 0);
  /* setup timer interval to 200 msecs */
  mod_timer(&my_timer, jiffies + msecs_to_jiffies(200));
  return 0;
}

void cleanup_module(void)
{
  /* remove kernel timer when unloading module */
  del_timer(&my_timer);
  return;
}
于 2012-06-19T09:37:40.437 回答
6

Linux 内核 4.15 版本前后,void setup_timer(timer, function, data); 变得过时,意图将其完全删除。

相反,现在我们必须使用

void timer_setup(
struct timer_list *timer, 
void (*callback)(struct timer_list *),
unsigned int flags
);

这可以在linux/timer.h文件中找到。

这是module_with_timer.c的完整示例

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/jiffies.h>
#include <linux/timer.h>

MODULE_LICENSE("GPL");

static struct timer_list my_timer;

void my_timer_callback(struct timer_list *timer) {
  printk(KERN_ALERT "This line is printed after 5 seconds.\n");
}

static int init_module_with_timer(void) {
  printk(KERN_ALERT "Initializing a module with timer.\n");

  /* Setup the timer for initial use. Look in linux/timer.h for this function */
  timer_setup(&my_timer, my_timer_callback, 0);
  mod_timer(&my_timer, jiffies + msecs_to_jiffies(5000));

  return 0;
}

static void exit_module_with_timer(void) {
  printk(KERN_ALERT "Goodbye, cruel world!\n");
  del_timer(&my_timer);
}

module_init(init_module_with_timer);
module_exit(exit_module_with_timer);

Makefile 是

obj-m = module_with_timer.o

# Get the current kernel version number
KVERSION = $(shell uname -r)

all:
    make -C /lib/modules/$(KVERSION)/build M=$(PWD) modules
clean:
    make -C /lib/modules/$(KVERSION)/build M=$(PWD) clean

注意:在实际编程中,最好检查我们正在编译的内核版本,然后使用 then 适当地启动计时器。

参考文献: https ://lwn.net/Articles/735887/

于 2020-09-01T00:15:18.963 回答
4

jiffies正如评论中所建议的那样,根据您确切想要做的事情,您可以直接使用来衡量时间。您还可以使用内核计时器,并且根据您问题中的信息,它们似乎更合适。

内核计时器 API 非常直观:

#include <linux/timer.h>
struct timer_list {
        /* ... */
        unsigned long expires;
        void (*function)(unsigned long);
        unsigned long data;
};

void init_timer(struct timer_list *timer);
struct timer_list TIMER_INITIALIZER(_function, _expires, _data);

void add_timer(struct timer_list * timer);
int del_timer(struct timer_list * timer);

所以你只需要定义一个定时器函数,然后初始化并启动定时器。

您有几个来源可以进一步了解此主题:

  • 了解 Linux 内核。这本书是内核的圣经。它在某些领域有些过时,但仍然是一个非常好的信息来源。
  • Linux 设备驱动程序。在开发设备驱动程序时,这是一本非常有用的书。这里也有在线版本。处理时间、计时器等的章节是第 7 章。这本书也可能有点过时,因为它也是 2005 年的。
  • Linux 内核开发。我没有查过这本书,但好处是它更新了很多(从 2010 年开始),所以与前两本书相比,你可能会发现一些更新的信息。
于 2012-06-19T09:37:52.190 回答