Linux驱动基础 | 延时函数的使用

目录

前言

思考

延时函数的定义

mdelay的定义

msleep的定义

延迟函数的使用

mdelay的使用

msleep的使用

注意事项

mdelay的注意事项

msleep的注意事项

总结

最后


前言

linux 驱动开发过程中,经常会用到延迟函数:udelaymdelayusleepmsleepusleep_range,所以本篇记录下获取内核延时所用到的API使用,用的比较多的mdelaymsleep。本篇讲下mdelaymsleep的使用

思考

msleepmdelay都是内核的ms级延时函数使用时有何区别,面试题经常会被问到,不会的同学看过来

延时函数的定义

mdelay的定义

延迟函数mdelay的定义位于file: include/linux/delay.h文件中

#ifndef mdelay
44 #define mdelay(n) (\
45         (__builtin_constant_p(n) && (n)<=MAX_UDELAY_MS) ? udelay((n)*1000) : \
46         ({unsigned long __ms=(n); while (__ms--) udelay(1000);}))
47 #endif

可以从定义看到,CPU在做忙等待,延时效果是精准延时

msleep的定义

延迟函数msleep的定义位于file: kernel/time/timer.c文件中

2029 /**
2030  * msleep - sleep safely even with waitqueue interruptions
2031  * @msecs: Time in milliseconds to sleep for
2032  */
2033 void msleep(unsigned int msecs)
2034 {      
2035         unsigned long timeout = msecs_to_jiffies(msecs) + 1;
2036 
2037         while (timeout)
2038                 timeout = schedule_timeout_uninterruptible(timeout);
2039 }      
2040 
2041 EXPORT_SYMBOL(msleep);

我们看下schedule_timeout_uninterruptible函数做了哪些事情

1914 signed long __sched schedule_timeout_uninterruptible(signed long timeout)
1915 {
1916         __set_current_state(TASK_UNINTERRUPTIBLE);
1917         return schedule_timeout(timeout);
1918 }
1919 EXPORT_SYMBOL(schedule_timeout_uninterruptible·);

解释下:__set_current_state(TASK_UNINTERRUPTIBLE)是设置当前进程设置不可中断状态, 就是让进程睡够指定时间才能唤醒,睡眠过程不可被中断(即UNINTERRUPTIBLE),意思不会被信号打断提前唤醒,然后schedule_timeout(timeout)函数timeout时间内主动调用schedule()主动让出CPU资源timeout时间,也就是说msleep会有上下文切换的开销,所以说msleep要比实际timeout的时间多延时一点时间

延迟函数的使用

我们就用mdelaymsleep都写个都延时1s,看看他们的延时效果

mdelay的使用

来看个mdelay的使用demo

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

static int __init my_module_init(void)
{
    printk(KERN_INFO "Module loading...\n");
    
    mdelay(1000); // 延迟 1000 毫秒 (1 秒)

    printk(KERN_INFO "1 second delay finished.\n");
    return 0;
}

static void __exit my_module_exit(void)
{
    printk(KERN_INFO "Module unloading...\n");
}

module_init(my_module_init);
module_exit(my_module_exit);

MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("A simple module using mdelay");
MODULE_AUTHOR("Your Name");

然后编译完放板子上加载驱动

图片

可以看到mdelay的实验结果是精准延时1s

msleep的使用

#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/delay.h>

static int __init my_module_init(void)
{
    printk(KERN_INFO "Module loading...\n");
    
    msleep(1000); // 延迟 1000 毫秒 (1 秒)

    printk(KERN_INFO "1 second delay finished.\n");
    return 0;
}

static void __exit my_module_exit(void)
{
    printk(KERN_INFO "Module unloading...\n");
}

module_init(my_module_init);
module_exit(my_module_exit);

MODULE_LICENSE("GPL");
MODULE_DESCRIPTION("A simple module using msleep");
MODULE_AUTHOR("Pan Shuai");

然后编译完放板子上加载驱动

图片

可以看到msleep的实验结果是延时略大于1s.

注意事项

mdelay的注意事项

  • 禁止中断:在某些上下文中(例如在中断上下文中使用时),可能会禁止中断,因此需要小心使用 mdelay
  • 延迟精度:比较适合精确的延迟

msleep的注意事项

  • 上下文msleep 不能在中断上下文中调用,因为中断上下文不能被挂起。应在进程上下文中使用。
  • 精度:由于任务调度的性质,msleep 可能无法保证非常精确的延迟

总结

1、msleep是休眠函数,它不涉及忙等待.时间是不准确的,比如msleep(200),大概率是会等待多于200ms的。适用于非实时场景

2、mdelay是忙等待函数,在延迟过程中无法运行其他任务.这个延迟的时间是准确的,所以适用于精准延时的时序场景

最后

本文学习到了msleepmdelay使用时有何区别。本篇文章学废的话,可以一键三连!欢迎关注公众号[Linux随笔录],不定期分享Linux小知识

<think>好的,用户想了解Linux内核驱动延时函数的实现方法或代码示例。我需要先回忆一下相关的内核API。记得常用的延时函数mdelay、udelay、ndelay,分别对应毫秒、微秒和纳秒级别的忙等待。这些函数在asm/delay.h或者linux/delay.h头文件中声明。不过要注意,udelay和ndelay在忙等期间会占用CPU,所以不能在原子上下文或中断处理中长时间使用。 还有msleep、usleep_range这些函数,它们属于可睡眠的延时,会让出CPU,适用于进程上下文。msleep在linux/delay.h中定义,而usleep_range可能更灵活一些,允许指定一个时间范围,有助于电源管理优化。 另外,高精度定时器hrtimer可能在某些情况下使用,特别是需要精确到纳秒级的延迟。不过这可能比较复杂,用户可能需要基础的例子。 需要检查用户提供的引用是否有相关内容。引用1提到了pinctrl的错误,可能不相关。引用2关于CPU热插拔的锁,引用3是无线驱动函数,似乎和延时无关。所以主要依靠自己的知识库。 应该分点说明不同的延时方法,强调使用场景和注意事项。比如忙等待和可睡眠的区别,以及不同函数的头文件和用法示例。最好给出代码片段,比如在驱动中如何使用mdelay和msleep。 还要注意内核版本的差异,比如较新的内核可能推荐usleep_range而不是msleep,因为后者可能不够精确或者有更好的替代。同时提醒用户避免在中断上下文中使用可能睡眠的函数,否则会导致内核崩溃。 最后,生成相关问题,帮助用户深入理解,比如如何选择延时函数,或处理长时间延迟的方法。确保回答结构清晰,符合用户要求的中文回答,使用正确的LaTeX格式,比如函数名用$msleep()$,并引用提供的引用,虽然这里可能用不上,但尽量关联。例如,在提到调度时可能引用CPU锁相关的引用2,但可能不太相关,所以可能不需要。</think>Linux内核驱动中实现延时的常用方法可分为忙等待和调度休眠两类,具体实现方式如下: ### 一、忙等待延时(原子操作) 1. **短延时函数**(适用于中断上下文) ```c #include <linux/delay.h> ndelay(nsecs); // 纳秒级延时 $n \in [1,10000]$ udelay(usecs); // 微秒级延时 $u \in [1,10000]$ mdelay(msecs); // 毫秒级延时 $m \in [1,1000]$ ``` *示例代码*: ```c udelay(200); // 200微秒延时 ``` 2. **实现原理**: 通过CPU空循环消耗时间,精度可达纳秒级但会完全占用CPU。其中`udelay()`基于BogoMIPS值计算循环次数[^2]。 ### 二、调度休眠延时(进程上下文) 1. **毫秒级休眠**: ```c #include <linux/delay.h> msleep(msecs); // 不可中断休眠 msleep_interruptible(msecs); // 可中断休眠 ``` 2. **微秒级精确休眠**: ```c usleep_range(min, max); // 推荐的最小误差实现 ``` *示例代码*: ```c usleep_range(5000, 10000); // 5-10毫秒延时 ``` ### 三、高精度定时器(hrtimer) ```c #include <linux/hrtimer.h> hrtimer_start(&timer, ktime_set(0, delay_ns), HRTIMER_MODE_REL); ``` 适用于需要纳秒级精度的场景,需配合定时器回调函数实现。 ### 四、注意事项 1. 在中断上下文中只能使用`ndelay()/udelay()` 2. `mdelay()`会触发`schedule()`检测,可能导致死锁 3. 超过1ms的延时应优先选择调度型函数 4. 驱动中推荐使用`usleep_range()`代替传统休眠函数 ```c // 典型使用场景示例 static irq_handler_t my_interrupt(int irq, void *dev_id) { udelay(50); // 中断处理中的短延时 return IRQ_HANDLED; } static int my_probe(...) { usleep_range(10000, 12000); // 设备初始化延时 msleep(100); // 长延时 } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值