当前位置:   article > 正文

Linux内核定时器

linux内核定时器

目录

一、时钟中断

二、延时机制

1. 短延迟:忙等待

2. 长延迟:忙等待

3. 睡眠延迟----阻塞类

三、定时器

(1)定义定时器结构体

(2)初始化定时器

(3)增加定时器 ------ 定时器开始计时

(4)删除定时器 -------定时器停止工作

(5)修改定时器

四、小练习—秒设备


一、时钟中断

硬件有一个时钟装置,该装置每隔一定时间发出一个时钟中断(称为一次时钟嘀嗒-tick),对应的中断处理程序就将全局变量jiffies_64加1

jiffies_64   是一个全局64位整型, jiffies全局变量为其低32位的全局变量,程序中一般用jiffies

HZ:可配置的宏,表示1秒钟产生的时钟中断次数,一般设为100或200



 

二、延时机制

1. 短延迟:忙等待

   ```c

   1. void ndelay(unsigned long nsecs)

   2. void udelay(unsigned long usecs)

   3. void mdelay(unsigned long msecs)

   ```

2. 长延迟:忙等待

   使用jiffies比较宏来实现

  1.    ```c
  2.    time_after(a,b)    //a > b
  3.    time_before(a,b)   //a < b
  4.    
  5.    //延迟100个jiffies
  6.    unsigned long delay = jiffies + 100;
  7.    while(time_before(jiffies,delay))
  8.    {
  9.        ;
  10.    }
  11.    
  12.    //延迟2s
  13.    unsigned long delay = jiffies + 2*HZ;
  14.    while(time_before(jiffies,delay))
  15.    {
  16.        ;
  17.    }
  18.    ```

3. 睡眠延迟----阻塞类

   ```c

   void msleep(unsigned int msecs);

   

   unsigned long msleep_interruptible(unsigned int msecs);

   ```


 

延时机制的选择原则:

1. 异常上下文中只能采用忙等待类

2. 任务上下文短延迟采用忙等待类,长延迟采用阻塞类

三、定时器

(1)定义定时器结构体

  1. ```c
  2. struct timer_list
  3. {
  4.    struct list_head entry;
  5.    unsigned long expires;  // 期望的时间值 jiffies + x * HZ
  6.    void (*function)(unsigned long); // 时间到达后,执行的回调函数,软中断异常上下文
  7.    unsigned long data;
  8. };
  9. ```

内核中用软中断实现所以是异常上下文不能被阻塞

(2)初始化定时器

```c

init_timer(struct timer_list *)

```

(3)增加定时器 ------ 定时器开始计时

```c

void add_timer(struct timer_list *timer);

```

(4)删除定时器 -------定时器停止工作

```c

int del_timer(struct timer_list * timer);

```

(5)修改定时器

```c

 int mod_timer(struct timer_list *timer, unsigned long expires);

```



 

  1. ```c
  2. 定义struct timer_list tl类型的变量
  3. init_timer(...);//模块入口函数
  4. //模块入口函数或open或希望定时器开始工作的地方
  5. tl.expires = jiffies + n * HZ //n秒
  6. tl.function = xxx_func;
  7. tl.data = ...;
  8. add_timer(....);
  9. //不想让定时器继续工作时
  10. del_timer(....);
  11. void xxx_func(unsigned long arg)
  12. {
  13.    ......
  14.    mod_timer(....);//如需要定时器继续隔指定时间再次调用本函数
  15. }
  16. ```

四、小练习—秒设备

这种设备一般不希望二次打开所以在定义结构体时加上只能一次打开机制。互斥锁的效率太低了。这里使用原子变量

error: implicit declaration of function ‘init_timer’; did you mean ‘init_timers’? [-Werror=implicit-function-declaration]



写完之后报了很多莫名其妙的错误,其中有一个是不认识init,我在网上查发现是因为4.15以后不支持这个函数了
 

error: ‘struct timer_list’ has no member named ‘data’




 

error: assignment from incompatible pointer type [-Werror=incompatible-pointer-types]
   pmydev->timer.function = timer_func;

 

 我在GIT上查到源码中这样定义结构体

 

初始化函数变成了

 __init_timer(_timer, _flags)

flags是一个标志我就写0了

 

error: macro "__init_timer" requires 3 arguments, but only 2 given 

 出现新的错误他说这个宏需要两个参数我只传了两个

算了哪天再来研究吧,使用开发板的内核编译就不会报错

  1. #include <linux/module.h>
  2. #include <linux/kernel.h>
  3. #include <linux/fs.h>
  4. #include <linux/cdev.h>
  5. #include <linux/uaccess.h>
  6. #include <linux/wait.h>
  7. #include <linux/sched.h>
  8. #include <linux/atomic.h>
  9. #include <linux/poll.h>
  10. #include <linux/device.h>
  11. #include <linux/timer.h>
  12. int major = 11;
  13. int minor = 0;
  14. int mysecond_num = 1;
  15. struct mysecond_dev
  16. {
  17. struct cdev mydev;
  18. int second;
  19. struct timer_list mytimer;
  20. /*Define atomic variables || 1 can open, 0 can not open*/
  21. atomic_t openflag;
  22. };
  23. struct mysecond_dev gmydev;
  24. void timer_func(unsigned long arg)
  25. {
  26. struct mysecond_dev *pmydev = (struct mysecond_dev *)arg;
  27. pmydev->second++;
  28. mod_timer(&pmydev->mytimer, jiffies + HZ * 1);
  29. }
  30. int mysecond_open(struct inode *pnode, struct file *pfile)
  31. {
  32. struct mysecond_dev *pmydev = NULL;
  33. pfile->private_data = (void *) (container_of(pnode->i_cdev, struct mysecond_dev, mydev));
  34. pmydev = (struct mysecond_dev *)pfile->private_data;
  35. if(atomic_dec_and_test(&pmydev->openflag))
  36. {
  37. pmydev->mytimer.expires = jiffies + HZ * 1;
  38. pmydev->mytimer.function = timer_func;
  39. pmydev->mytimer.data = (unsigned long)pmydev;
  40. add_timer(&pmydev->mytimer);
  41. return 0;
  42. }
  43. else
  44. {
  45. atomic_inc(&pmydev->openflag);
  46. printk("The device is opened already\n");
  47. return -1;
  48. }
  49. return 0;
  50. }
  51. int mysecond_close(struct inode *pnode, struct file *pfile)
  52. {
  53. //printk("mysecond_close\n");
  54. /*C90 requires printk after the variable declaration*/
  55. struct mysecond_dev *pmydev = (struct mysecond_dev *)pfile->private_data;
  56. del_timer(&pmydev->mytimer);
  57. atomic_set(&pmydev->openflag,1);
  58. return 0;
  59. }
  60. ssize_t mysecond_read(struct file *pfile, char __user *puser, size_t size, loff_t *p_pos)
  61. {
  62. struct mysecond_dev *pmydev = (struct mysecond_dev *)pfile->private_data;
  63. int ret = 0;
  64. if(size < sizeof(int))
  65. {
  66. printk("the expect read size is invalid\n");
  67. return -1;
  68. }
  69. if(size >= sizeof(int))
  70. {
  71. size = sizeof(int);
  72. }
  73. ret = copy_to_user(puser, &pmydev->second, size);
  74. if(ret)
  75. {
  76. printk("copy to user failed\n");
  77. return -1;
  78. }
  79. return size;
  80. }
  81. struct file_operations myops = {
  82. .owner = THIS_MODULE,
  83. .open = mysecond_open,
  84. .release = mysecond_close,
  85. .read = mysecond_read,
  86. };
  87. int __init mysecond_init(void)
  88. {
  89. int ret = 0;
  90. dev_t devno = MKDEV(major,minor);
  91. /*Apply for device number*/
  92. ret = register_chrdev_region(devno, mysecond_num, "mysecond");
  93. if(ret)
  94. {
  95. ret = alloc_chrdev_region(&devno, minor, mysecond_num, "mysecond");
  96. if(ret)
  97. {
  98. printk("get devno failed\n");
  99. return -1;
  100. }
  101. major = MAJOR(devno);//Easy to miss *****
  102. }
  103. /*Assign the 'struct cdev' a set of operation functions*/
  104. cdev_init(&gmydev.mydev, &myops);
  105. /*Add 'struct cdev' to the kernel's data structure*/
  106. gmydev.mydev.owner = THIS_MODULE;
  107. cdev_add(&gmydev.mydev, devno, mysecond_num);//add to Hash.
  108. init_timer(&gmydev.mytimer);
  109. /*initialize the atomic variable to 1*/
  110. atomic_set(&gmydev.openflag,1);
  111. return 0;
  112. }
  113. void __exit mysecond_exit(void)
  114. {
  115. dev_t devno = MKDEV(major,minor);
  116. cdev_del(&gmydev.mydev);
  117. //printk("mysecond will exit\n");
  118. unregister_chrdev_region(devno, mysecond_num);
  119. }
  120. MODULE_LICENSE("GPL");
  121. module_init(mysecond_init);
  122. module_exit(mysecond_exit);

 

  1. #include <sys/types.h>
  2. #include <sys/stat.h>
  3. #include <fcntl.h>
  4. #include <unistd.h>
  5. #include <stdio.h>
  6. int main(int argc,char *argv[])
  7. {
  8. int fd = -1;
  9. int sec = 0;
  10. if(argc < 2)
  11. {
  12. printf("The argument is too few\n");
  13. return 1;
  14. }
  15. fd = open(argv[1],O_RDONLY);
  16. if(fd < 0)
  17. {
  18. printf("open %s failed\n",argv[1]);
  19. return 2;
  20. }
  21. sleep(3);
  22. read(fd,&sec,sizeof(sec));
  23. printf("The second is %d\n",sec);
  24. close(fd);
  25. fd = -1;
  26. return 0;
  27. }

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/article/detail/37895
推荐阅读
相关标签
  

闽ICP备14008679号