0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Linux内核用户态是如何睡眠的

B4Pb_gh_6fde77c 来源:Linux内核远航者 作者:Linux内核远航者 2021-08-16 15:06 次阅读

4.用户态睡眠

以sleep为例来说明任务在用户态是如何睡眠的。

首先我们通过strace工具来看下其调用的系统调用:

$ strace sleep 1

...

close(3) = 0

clock_nanosleep(CLOCK_REALTIME, 0, {tv_sec=1, tv_nsec=0}, NULL) = 0

close(1) = 0

...

可以发现sleep主要调用clock_nanosleep系统调用来进行睡眠(也就是说用户态任务睡眠需要调用系统调用陷入内核)。

下面我们来研究下clock_nanosleep的实现(这里集中到睡眠的实现,先忽略掉定时器等诸多的技术细节):

kernel/time/posix-timers.c

SYSCALL_DEFINE4(clock_nanosleep

-》const struct k_clock *kc = clockid_to_kclock(which_clock); //根据时钟类型得到内核时钟结构

return kc-》nsleep(which_clock, flags, &t); //调用内核时钟结构的nsleep回调

我们传递过来的时钟类型为CLOCK_REALTIME,则调用链为:

kc-》nsleep(CLOCK_REALTIME, flags, &t)

-》clock_realtime.nsleep

-》common_nsleep

-》hrtimer_nanosleep //kernel/time/hrtimer.c

-》hrtimer_init_sleeper_on_stack

-》__hrtimer_init_sleeper

-》__hrtimer_init(&sl-》timer, clock_id, mode); //初始化高精度定时器

sl-》timer.function = hrtimer_wakeup; //设置超时回调函数

sl-》task = current;。//设置超时时要唤醒的任务

-》do_nanosleep //睡眠操作

可以看到,睡眠函数最终调用到hrtimer_nanosleep,它调用了两个主要函数:__hrtimer_init_sleeper和do_nanosleep,前者主要设置高精度定时器,后者就是真正的睡眠,主要来看下do_nanosleep:

kernel/time/hrtimer.c

do_nanosleep

-》

do {

set_current_state(TASK_INTERRUPTIBLE); //设置可中断的睡眠状态

hrtimer_sleeper_start_expires(t, mode); //开启高精度定时器

if (likely(t-》task))

freezable_schedule(); //主动调度

hrtimer_cancel(&t-》timer);

mode = HRTIMER_MODE_ABS;

} while (t-》task && !signal_pending(current)); //是否记录的有任务且没有挂起的信号

__set_current_state(TASK_RUNNING); //设置为可运行状态

do_nanosleep函数是睡眠的核心实现:首先设置任务的状态为可中断的睡眠状态,然后开启了之前设置的高精度定时器,随即调用freezable_schedule进行真正的睡眠。

来看下freezable_schedule:

//include/linux/freezer.h

freezable_schedule

-》schedule()

-》__schedule(false);

可以看到最终调用主调度器__schedule进行主动调度。

当任务睡眠完成,定时器超时,会调用之前在__hrtimer_init_sleeper设置的超时回调函数hrtimer_wakeup将睡眠的任务唤醒(关于进程唤醒在这里就不在赘述,在后面的进程唤醒专题文章在进行详细解读),然后就可以再次获得处理器的使用权了。

总结:处于用户态的任务,如果想要睡眠一段时间必须向内核请求服务(如调用clock_nanosleep系统调用),内核中会设置一个高精度定时器,来记录要睡眠的任务,然后设置任务状态为可中断的睡眠状态,紧接着发生主动调度,这样任务就发生睡眠了。

5.内核态睡眠

当任务处于内核态时,有时候也需要睡眠一段时间,不像任务处于用户态需要发生系统调用来请求内核进行睡眠,在内核态可以直接调用睡眠函数。当然,内核态中,睡眠有两种场景:一种是睡眠特定的时间的延迟操作(唤醒条件为超时),一种是等待特定条件满足(如IO读写完成,可睡眠的锁被释放等)。

下面分别以msleep和mutex锁为例讲解内核态睡眠:

5.1 msleep

msleep做ms级别的睡眠延迟。

//kernel/time/timer.c

void msleep(unsigned int msecs)

{

unsigned long timeout = msecs_to_jiffies(msecs) + 1; //ms时间转换为jiffies

while (timeout)

timeout = schedule_timeout_uninterruptible(timeout); //不可中断睡眠

}

下面看下schedule_timeout_uninterruptible:

这里涉及到一个重要数据结构process_timer

struct process_timer {

struct timer_list timer; //定时器结构

struct task_struct *task; //定时器到期要唤醒的任务

};

schedule_timeout_uninterruptible

-》 __set_current_state(TASK_UNINTERRUPTIBLE); //设置任务状态为不可中断睡眠

return schedule_timeout(timeout);

-》expire = timeout + jiffies; //计算到期时的jiffies值

timer.task = current; //记录定时器到期要唤醒的任务 为当前任务

timer_setup_on_stack(&timer.timer, process_timeout, 0); //初始化定时器 超时回调为process_timeout

__mod_timer(&timer.timer, expire, MOD_TIMER_NOTPENDING); //添加定时器

schedule(); //主动调度

再看下超时回调为process_timeout:

process_timeout

-》struct process_timer *timeout = from_timer(timeout, t, timer); //通过定时器结构获得process_timer

wake_up_process(timeout-》task); //唤醒其管理的任务

可以看到,msleep实现睡眠也是通过定时器,首先设置当前任务状态为不可中断睡眠,然后设置定时器超时时间为传递的ms级延迟转换的jiffies,超时回调为process_timeout,然后将定时器添加到系统中,最后调用schedule发起主动调度,当定时器超时的时候调用process_timeout来唤醒睡眠的任务。

5.2 mutex锁

mutex锁是可睡眠锁的一种,当申请mutex锁时发现其他内核路径已经持有这把锁,当前任务就会睡眠等待在这把锁上。

下面我们来看他的实现,主要看睡眠的部分:

kernel/locking/mutex.c

mutex_lock

-》__mutex_lock_slowpath

-》__mutex_lock(lock, TASK_UNINTERRUPTIBLE, 0, NULL, _RET_IP_) //睡眠的状态为不可中断睡眠

-》__mutex_lock_common

-》

...

waiter.task = current; //记录需要唤醒的任务为当前任务

set_current_state(state); //设置睡眠状态

for(;;) {

if (__mutex_trylock(lock)) //尝试获得锁

goto acquired;

schedule_preempt_disabled();

-》schedule(); //主动调度

}

acquired:

__set_current_state(TASK_RUNNING);//设置状态为可运行状态

可以看到mutex锁实现睡眠套路和之前是一样的:申请mutex锁的时候,如果其他内核路径已经持有这把锁,首先通过mutex锁的相关结构来记录下当前任务,然后设置任务状态为不可中断睡眠,接着在一个for循环中调用schedule_preempt_disabled发生主动调度,于是当前任务就睡眠在这把锁上。

当其他内核路径释放了这把锁,就会唤醒等待在这把锁上的任务,当前任务就获得了这把锁,然后进入锁的临界区,唤醒操作就完成了(关于唤醒的技术细节,后面的唤醒专题会详细讲解)。

6.总结

进程睡眠按照应用场景可以分为:延迟睡眠和等待某些特定条件而睡眠,实际上都可以归于等待某些特定条件而睡眠,因为延迟特定时间也可以作为特定条件。

进程睡眠按照进程所处的特权级别可以分为:用户态进程睡眠和内核态进程睡眠,用户态进程睡眠需要进程通过系统调用陷入内核来发起睡眠请求。对于进程睡眠,内核主要需要做三大步操作:

1.设置任务状态为睡眠状态 2.记录睡眠的任务 3.发起主动调度。这三大步操作都是非常有必要,第一步设置睡眠状态为后面调用主调度器做必要的标识准备;第二步记录下睡眠的任务是为了以后唤醒任务来准备的;第三步是睡眠的主体部分,这里会将睡眠的任务从运行队列中踢出,选择下一个任务运行。

原文标题:深入理解Linux内核之进程睡眠(下)

文章出处:【微信公众号:Linux阅码场】欢迎添加关注!文章转载请注明出处。

责任编辑:haq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表德赢Vwin官网 网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 内核
    +关注

    关注

    3

    文章

    1332

    浏览量

    40042
  • Linux
    +关注

    关注

    87

    文章

    11094

    浏览量

    207651

原文标题:深入理解Linux内核之进程睡眠(下)

文章出处:【微信号:gh_6fde77c41971,微信公众号:FPGA干货】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    Linux内核测试技术

    Linux 内核Linux操作系统的核心部分,负责管理硬件资源和提供系统调用接口。随着 Linux 内核的不断发展和更新,其复杂性和代码规
    的头像 发表于08-13 13:42 164次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>内核</b>测试技术

    Linux内核中的页面分配机制

    Linux 内核中是如何分配出页面的,如果我们站在CPU的角度去看这个问题,CPU能分配出来的页面是以物理页面为单位的。也就是我们计算机中常讲的分页机制。本文就看下 Linux 内核是如何管
    的头像 发表于08-07 15:51 117次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>内核</b>中的页面分配机制

    欢创播报 华为宣布鸿蒙内核已超越Linux内核

    1 华为宣布鸿蒙 内核已超越 Linux 内核6月21日,在华为开发者大会上, HarmonyOS NEXT(鸿蒙NEXT)——真正独立于安卓和iOS的鸿蒙操作系统,正式登场。这是HarmonyOS
    的头像 发表于06-27 11:30 517次阅读

    使用 PREEMPT_RT 在 Ubuntu 中构建实时Linux内核

    盟通技术干货构建实时 Linux 内核简介盟通技术干货Motrotech如果需要在 Linux中实现实时计算性能,进而有效地将 Linux转变为RTOS,那么大多数发行版都可以打上名为PREE
    的头像 发表于04-12 08:36 1058次阅读
    使用 PREEMPT_RT 在 Ubuntu 中构建实时 <b class='flag-5'>Linux</b> <b class='flag-5'>内核</b>

    获取Linux内核源码的方法

    (ELF1/ELF1S开发板及显示屏) Linux 内核是操作系统中最核心的部分,它负责管理计算机硬件资源,并提供对应用程序和其他系统组件的访问接口,控制着计算机的内存、处理器、设备驱动程序和文件系统等
    的头像 发表于12-13 09:49 475次阅读
    获取<b class='flag-5'>Linux</b><b class='flag-5'>内核</b>源码的方法

    Linux内核自解压过程分析

    uboot完成系统引导以后,执行环境变量bootm中的命令;即,将 Linux 内核调入内存中并调用do_bootm函数启动 内核,跳转至kernel的起始位置。
    的头像 发表于12-08 14:00 658次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>内核</b>自解压过程分析

    Linux读写锁逻辑解析—Linux为何会引入读写锁?

    除了mutex,在 linux 内核中,还有一个经常用到的 睡眠锁就是rw semaphore(后文简称为rwsem),它到底和mutex有什么不同呢?
    的头像 发表于12-04 11:04 675次阅读
    <b class='flag-5'>Linux</b>读写锁逻辑解析—<b class='flag-5'>Linux</b>为何会引入读写锁?

    Linux内核UDP收包为什么效率低

    包效率真的很低,这是为什么?有没有办法去尝试着优化?而不是动不动就DPDK。 我们从最开始说起。 Linux 内核作为一个通用操作系统 内核,脱胎于UNIX那一套现代操作系统理论。 但一开始不知道怎么回事将网络协议栈的实现塞进了
    的头像 发表于11-13 10:38 364次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>内核</b>UDP收包为什么效率低

    如何优化Linux内核UDP收包效率低

    真的很低,这是为什么?有没有办法去尝试着优化?而不是动不动就DPDK。 我们从最开始说起。 Linux 内核作为一个通用操作系统 内核,脱胎于UNIX那一套现代操作系统理论。 但一开始不知道怎么回事将网络协议栈的实现塞进了
    的头像 发表于11-10 10:51 421次阅读
    如何优化<b class='flag-5'>Linux</b><b class='flag-5'>内核</b>UDP收包效率低

    内核?还是用户?哪一个更适合TCP/IP协议栈呢?

    最近一段时间,我几乎每天深夜都在做一件事,对比mtcp, Linux 内核协议栈的收包处理和TCP新建连接的性能,同时还了解了一下腾讯的F-Stack。这里指明,我的mtcp使用的是netmap作为底层支撑,而不是DPDK。
    的头像 发表于11-03 09:16 446次阅读
    <b class='flag-5'>内核</b><b class='flag-5'>态</b>?还是<b class='flag-5'>用户</b><b class='flag-5'>态</b>?哪一个更适合TCP/IP协议栈呢?

    RT-Thread Smart qemu-virt64-riscv用户userapps的编译与运行

    rt-smart 上,为了实现 用户 内核 的分离,使用了【系统调用】,这个系统调用可以认为是个 sdk
    的头像 发表于10-08 15:33 767次阅读
    RT-Thread Smart qemu-virt64-riscv<b class='flag-5'>用户</b><b class='flag-5'>态</b>userapps的编译与运行

    Linux内核编译脚本

    获得编译命令及选项 编译 linux时,默认不会显示编译的命令,如果你要获得编译命令及其选项,可以在make命令后面加上宏定义: make V= 1 如果希望编译系统告诉你为何某个目标文件需要重新
    的头像 发表于09-27 11:52 514次阅读

    Linux电源管理的组成与睡眠唤醒

    (Hibernate)、 睡眠(Sleep,在 Kernel 中称作Suspend)。在 内核中,大致可以分为三个软件层次: API Layer,用于向 用户空间提供接口,其中关机和重启的接口形式是系统调用,Hibernat
    的头像 发表于09-11 15:54 423次阅读
    <b class='flag-5'>Linux</b>电源管理的组成与<b class='flag-5'>睡眠</b>唤醒

    linux内核源代码详解

     在安装好的 Linux系统中, 内核的源代码位于/ust/src/ linux.如果是从GNU网站下载的 Linux 内核的tar文件,则展开以后在
    发表于09-06 17:01 4次下载

    Linux内核如何使用结构体和函数指针?

    我将结合具体的 Linux 内核驱动框架代码来展示 Linux 内核如何使用结构体和函数指针。
    的头像 发表于09-06 14:17 773次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>内核</b>如何使用结构体和函数指针?