作者:云天明
链接:https://www.zhihu.com/question/50764115/answer/122613742
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

cpu里面有可编程间隔定时器(PIT,Programmable interval timer),目前x86-64/arm/8051-based的绝大多数cpu/mcu都是内置PIT的,PIT以一个可调节的时间间隔触发时钟中断,使得操作系统的时钟中断处理程序可以可调节地周期性运行。时钟中断处理程序负责维护所有的软件定时器,在当前进程的时间片用光,或有定时器触发时执行进程调度(线程调度),并维护软件时钟。

PIT和RTC的区别:计算机中有一枚纽扣电池负责在关机时维持CMOS数据并驱动一个低频时钟,这个时钟叫做实时钟(Real-time clock),注意RTC是相当不精确的,所以大多数操作系统通常在开机之后维护一个软件时钟,通过时钟中断和网咯时间协议(http://www.ntp.org/)维护的软件时钟比RTC精确和可靠得多,绝大多数操作系统除了开机时会读取RTC,关机时将RTC与软件时钟同步以外,大多数时间RTC都是孤独地运行的。

PIT与WDT的区别:有些嵌入式MCU除了提供PIT以外还提供看门狗定时器(Watchdog timer),有些PIT可以用于WDT功能,有些WDT具有PIT功能,有些场合PIT和WDT可能使用相同的时钟源,但两者是不同的。PIT用于周期性产生中断信号,而WDT一旦触发,产生的是复位信号,嵌入式操作系统需要不停地重置WDT以阻止复位信号的产生,WDT主要用于从软件错误中自动恢复。

嗯,实名反对另外…………不是咱不懂别强答好么←_←

以上是硬件部分

当一个线程执行了sleep(1000)时,首先会陷入到内核态执行系统调用处理程序,相关的处理程序会设置一个软件定时器(1000),然后转向线程调度,这时候会进行线程上下文切换和进程上下文切换,线程调度程序将该线程标记为等待态,并标记该线程等待该定时器的信号,然后执行线程调度。

当时钟中断处理程序发现这个软件定时器触发时,便将这个信号量置位(或者复位?),然后通知线程调度模块,线程调度发现这个信号量与某个线程相关联,于是将线程标记为就绪态,这时候
{当前线程的优先级更高:继续执行当前进程,
当前进程是idle/当前线程优先级低:转向执行刚进入就绪态的线程并标记为当前线程,切换进程上下文,恢复保存的线程现场,然后把触发的定时器和信号量参数传递给线程的内核态部分(系统调用处理程序),系统调用处理程序删除定时器然后返回到用户态}

sleep存在误差的原因在于,你交出控制权以后,定时的时间到了你还能不能被分到时间片是未知数,所以有可能定时器时间到了线程并没有马上进入执行态,但是这个在pc上不是个事,服务器才需要考虑这个

不过linux内核有个奇葩的东西叫做高解析度定时模式,和另外一个奇葩的东西叫做完全无滴答模式,在高解析度定时模式下,linux通过对PIT编程使得定时尽量精确(PIT的触发周期是可调的,最低我记得小于微秒级,和CPU有关),这样使得软件定时器可以精确到微秒级,有时候在高精度定时模式下linux的时钟中断是完全没有周期性的,每触发一次时钟中断就要改一次定时周期。完全无滴答模式是把PIT直接关掉的,当然如果你有定时需求系统会把这玩意再打开。

linux 时间模块 一的更多相关文章

  1. linux 时间模块 三

    LINUX的时钟中断中涉及至二个全局变量一个是xtime,另一个则是jiffies.有一个与时间有关的时钟:实时时钟(RTC),这是一个硬件时钟,用来持久存放系统时间,系统关闭后靠主板上的微型电池保持 ...

  2. linux 时间模块 二

    Linux中有硬件时钟与系统时钟 在Linux中有硬件时钟与系统时钟等两种时钟.硬件时钟是指主机板上的时钟设备,也就是通常可在BIOS画面设定的时钟.系统时钟则是指kernel中的时钟.当Linux启 ...

  3. s14 第5天 时间模块 随机模块 String模块 shutil模块(文件操作) 文件压缩(zipfile和tarfile)shelve模块 XML模块 ConfigParser配置文件操作模块 hashlib散列模块 Subprocess模块(调用shell) logging模块 正则表达式模块 r字符串和转译

    时间模块 time datatime time.clock(2.7) time.process_time(3.3) 测量处理器运算时间,不包括sleep时间 time.altzone 返回与UTC时间 ...

  4. Linux时间子系统之(二):软件架构

    专题文档汇总目录 Notes:从框架上讲解了时间子系统,从底向上包括CPU Local TImer.Global Counter.Clock Souce/Clock Events模块管理.Tick D ...

  5. Linux时间子系统之(三):用户空间接口函数

    专题文档汇总目录 Notes:用户空间时间相关接口函数: 类型 API 精度 说明 时间 time stime time_t 精度为秒级 逐渐要被淘汰.需要定义__ARCH_WANT_SYS_TIME ...

  6. Linux时间子系统之(四):timekeeping

    专题文档汇总目录 Notes:timekeeping模块的狠心数据结构是timekeeper,它维护了系统不同类型时钟的时间值,并且介绍了获取不同类型时钟时间的函数. clocksource切换通过c ...

  7. Linux时间子系统之(五):POSIX Clock

    专题文档汇总目录 Notes: 本章主要介绍了若干种类的静态时钟,这些时钟都可以通过k_clock表示,注册到posix_clocks中.这些都是静态时钟,可以分为三大类:各种REALTIME时钟.带 ...

  8. Linux时间子系统之(十三):Tick Device layer综述

    专题文档汇总目录 Notes:从概念层次描述了tick-comm.oneshot tick.broadcast tick:重点介绍了tick和tickless概念及其区别,两种tick device: ...

  9. Linux时间子系统之(十四):tick broadcast framework

    专题文档汇总目录 Notes:BroadcastTick作为cpuidle的waker,硬件基础.BroadcastTick嵌入在当前系统Tick框架中.BroadcastTick设备初始化:周期性T ...

随机推荐

  1. MySQL中一条SQL的加锁分析

    MySQL中一条SQL的加锁分析 id主键 + RC id唯一索引 + RC id非唯一索引 + RC id无索引 + RC id主键 + RR id唯一索引 + RR id非唯一索引 + RR id ...

  2. Golang 简单 http 代理转发

    程序基本实现了对http的完整转发,目前暂不支持https windows需要在设置中的网络>代理设置为手动,并开启代理服务器,填写ip和端口 // httpForward package ma ...

  3. VUE2.0声明周期钩子:不同阶段不同钩子的开启

  4. 转载:将画布(canvas)图像保存成本地图片的方法

    之前我曾介绍过如何将HTML5画布(canvas)内容转变成图片形式,方法十分简单.但后来我发现只将canvas内容转变成图片输出还不够,如何能将转变后的图片保存到本地呢? 其实,这个方法也是非常简单 ...

  5. laravel中redis各方法的使用

    在laravel中使用redis自带方法的时候会发现许多原生的方法都不存在了,laravel对其进行了重新的封装但是在文档中并没有找到相关的资料最后在 \vendor\predis\predis\sr ...

  6. k短路模板

    https://acm.taifua.com/archives/jsk31445.html 链接: https://nanti.jisuanke.com/t/31445 #include <io ...

  7. Docker容器技术的核心原理

    目录 1 前言 2 docker容器技术 2.1 隔离:Namespace 2.2 限制:Cgroup 2.3 rootfs 2.4 镜像分层 3 docker容器与虚拟机的对比 1 前言 上图是百度 ...

  8. POJ:2109-Power of Cryptography(关于double的误差)

    Power of Cryptography Time Limit: 1000MS Memory Limit: 30000K Description Current work in cryptograp ...

  9. 27、android log日志

    一.记住 加写sd卡权限 二.代码 package com.example.logtest; import java.io.File; import java.io.IOException; impo ...

  10. Python socket粘包问题(初级解决办法)

    server端配置: import socket,subprocess,struct from socket import * server=socket(AF_INET,SOCK_STREAM) s ...