Linux/Uinx 系统编程:定时器以及时钟同步

  • 本章讨论了定时器和定时器服务;介绍了硬件定时器的原理和基于Intel x86 的PC中的硬件定时器;讲解了CPU操作和中断处理;描述了Linux中与定时器相关的系统调用、库函数和定时器服务命令;探讨了进程间隔定时器、定时器生成的信号,并通过示例演示了进程间隔定时器。编程项目的目的是要在一个多任务处理系统中实现定时器、定时器中断和间隔定时器。

    多任务处理系统作为一个Linux 进程运行,该系统是Linux进程内并发任务的一个虚拟CPU。Linux 进程的实时模式间隔定时器被设计为定期生成 SIGALRM信号,充当虚拟CPU的定时器中断,虚拟CPU使用SIGALRM信号捕捉器作为定时器的中断处理程序。

硬件定时器

定时器是由时钟源和可编程计数器组成的硬件设备。时钟源通常是一个晶体振荡器,会产生周期性电信号,以精确的频率驱动计数器。使用一个倒计时值对计数器进行编程,每个时钟信号减1。当计数减为0时,计数器向CPU生成一个定时器中断、将计数值重新加载到计数器中,并重复倒计时。计数器周期称为定时器刻度,是系统的基本计时单元。

个人计算机定时器

基于Intel x86的个人计算机有数个定时器(Bovet 和 Cesati 2005)。

  1. 实时时钟(RTC)∶ RTC由一个小型备用电池供电。即使在个人计算机关机时,它也能连续运行。它用于实时提供时间和日期信息。当Linux 启动时,它使用 RTC更新系统时间变量,以与当前时间保持一致。在所有类Unix 系统中,时间变量是一个长整数,包含从1970年1月1日起经过的秒数。
  2. 可编程间隔定时器(PIT)(Wang 2015)∶PIT是与CPU分离的一个硬件定时器。可对它进行编程,以提供以毫秒为单位的定时器刻度。在所有I/O设备中,PIT可以最高优先级IRQ0 中断。PIT定时器中断由Linux 内核的定时器中断处理程序来处理,为系统操作提供基本的定时单元,例如进程调度、进程间隔定时器和其他许多定时事件。
  3. 多核CPU中的本地定时器((Intel 1997;Wang 2015)∶在多核 CPU中,每个核都是一个独立的处理器、它有自己的本地定时器,由 CPU 时钟驱动。
  4. 高分辨率定时器;大多数电脑都有一个时间戳定时器(TSC),由系统时钟驱动。它的内容可通过 64.位 TSC寄存器读取。由干不同系统主板的时钟频率可能不同、TSC 不活合作为实时设备,但它可提供纳秒级的定时器分辨率。一些高端个人计算机可能还配备有专用高速定时器,以提供纳秒级定时器分辨率。

CPU操作

每个CPU 都有一个程序计数器(PC),也称为指令指针(IP),以及一个标志或状态寄存器(SR)、一个堆栈指针(SP)和几个通用寄存器,当PC指向内存中要执行的下一条指令时,SR 包含 CPU 的当前状态,如操作模式、中断掩码和条件码,SP指向当前堆栈栈顶。

堆栈是CPU用于特殊操作(如 push、pop调用和返回等)的一个内存区域。CPU操作可通过无限循环进行建模。

由于无效地址、非法指令、越权等原因,可能会出现一个错误状态,称为异常或陷阱。当CPU遇到异常时,它会根据内存中预先安装的指针来执行软件中的异常处理程序。

在每条指令执行结束时,CPU会检查挂起的中断。中断是 I/O设备或协处理器发送给CPU的外部信号,请求CPU服务。如果有挂起的中断请求,但是CPU未处于接受中断的状态,即它的状态寄存器已经屏蔽了中断,CPU会忽略中断请求。继续执行下二条指令。

否则,它将直接执行中断处理。在中断处理结束时,它将恢复指令的正常执行。中断处理和异常处理都在操作系统内核中进行。在大多数情况下,用户级程序无法访问它们,但它们是理解操作系统(如 Linux)定时器服务和信号的关键。

中断处理

外部设备(如定时器)的中断被馈送到中断控制器的预定义输入行(Intel 1990;Wang 2015),按优先级对中断输入排序,并将具有最高优先级的中断作为中断请求(IRQ)路由到 CPU。

在每条指令执行结束时,如果CPU未处于接受中断的状态,即在 CPU 的状态寄存器中屏蔽了中断,它将忽略中断请求,使其处于挂起状态,并继续执行下一条指令。

如果CPU处于接受中断状态,即中断未被屏蔽,那么 CPU将会转移它正常的执行顺序来进行中断处理。对于每个中断,可以编程中断控制器以生成一个唯一编号,叫作中断向量、标识中断源。

在获取中断向量号后,CPU用它作为内存中中断向量表(AMD64)中的条目索引,条目包含一个指向中断处理程序人口地址的指针来实际处理中断。当中断处理结束时,CPU恢复指令的正常执行。

时钟服务函数

在几乎所有的操作系统(OS)中,操作系统内核都会提供与时钟相关的各种服务。时钟服务可通过系统调用、库函数和用户级命令调用。包括**gettimeofday-settimeofday系统调用、timetimes系统调用、timedate**命令。

gettimeofday系统调用

  1. 简介:在C语言中可以使用函数gettimeofday来得到时间。它的精度可以达到微秒。
  2. 函数原型:
#include<sys/time.h>
int gettimeofday(struct timeval*tv,struct timezone *tz)
  1. 说明:gettimeofday会把目前的时间用tv结构体返回,当地时区的信息则放到tz所指的结构中。

  2. 结构体:

    • timeval
    struct timeval {long tv_sec; /*秒*/long tv_usec; /*微妙*/
    };
    • timezone
    struct timezone {int tz_minuteswest; /*和greenwich时间差了多少分钟*/int tz_dsttime; /*type of DST correction*/
    };

settimeofday系统调用

  1. 简介:settimeofday函数用于设置当前时间。
  2. 函数原型:
#include<sys/time.h>
int settimeofday(const struct timeval *tv , const struct timezone *tz);
  1. 说明:settimeofday会设置当前时间,时间由tv指向的结构体提供,时区由tz指向的结构体提供。
  2. 结构体:同gettimeofday函数。

注意:这两个函数需要root权限才能使用。

示例代码:

#include <stdio.h>
#include <sys/time.h>
#include <stdlib.h>int main()
{float time_use=0;struct timeval start;struct timeval end;gettimeofday(&start,NULL); //gettimeofday(&start,&tz);结果一样printf("start.tv_sec:%d\n",start.tv_sec);printf("start.tv_usec:%d\n",start.tv_usec);int i = 100000000;while(i--);gettimeofday(&end,NULL);printf("end.tv_sec:%d\n",end.tv_sec);printf("end.tv_usec:%d\n",end.tv_usec);time_use=(end.tv_sec-start.tv_sec)*1000000+(end.tv_usec-start.tv_usec);//微秒printf("time_use is %.10f\n",time_use);
}

间隔定时器

Linux为每个进程提供了三种不同类型的间隔计时器,可用作进程计时的虚拟时钟。间隔定时器由setitimer()系统调用创建。getitimer()系统调用返回间隔定时器的状态。

int getitimer(int which,struct itimerval *curr_value);
int setitimer(int which,const struct itimerval *new_value,
struct itimerval *old_value);

各间隔定时器在参数which指定的不同时间域中工作。当间隔定时器定时到期时,会向进程发送一个信号,并将定时器重置为指定的间隔值(如果是非零数)。一个信号就是发送给某个进程进行处理的一个数字(1到31)。

有3类间隔定时器,分别是:

  1. ITIMER_REAL:实时减少,在到期时生成一个SIGALRM(14)信号。
  2. ITIMER_VIRTUAL:仅当进程在用户模式下执行时减少,在到期时生成一个SIGVTALRM(26)信号。
  3. ITIMER_PROF:当进程正在用户模式和系统(内核)模式下执行时减少。这类间隔定时器与**ITIMER_VIRTUAL**结合使用,通常用于分析应用程序在用户模式和内核模式下花费的时间。它在到期时生成一个SIGPROF(27)信号。

具体有关信号的内容,将来会发布在同专栏系列中。

间隔定时器的值用以下结构体(在<sys/time.h>中定义):

struct itimerval {struct timeval it_interval;  // 下次定时器到期的时间间隔struct timeval it_value;     // 定时器首次到期的时间
};struct timeval {time_t      tv_sec;          // 秒suseconds_t tv_usec;         // 微秒
};

其中,it_value表示定时器首次到达定时时间的时长。如果此值为0,则定时器将被禁用it_interval表示定时器到期后,再次到期的时间间隔。如果此值为0,则定时器只会在it_value所指定的时间后触发一次。如果it_interval不为0,则定时器将会每隔it_interval所指定的时间触发一次。

这两个字段都是timeval结构体,包含两个成员:tv_sectv_usectv_sec表示秒数,tv_usec表示微秒数。这两个字段共同指定了时间长度。例如,如果tv_sec为2,tv_usec为500000,那么总的时间长度就是2.5秒。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/760828.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C语言经典算法-6

文章目录 其他经典例题跳转链接31.数字拆解32.得分排行33.选择、插入、气泡排序34.Shell 排序法 - 改良的插入排序35.Shaker 排序法 - 改良的气泡排序 其他经典例题跳转链接 C语言经典算法-1 1.汉若塔 2. 费式数列 3. 巴斯卡三角形 4. 三色棋 5. 老鼠走迷官&#xff08;一&…

《C语言深度剖析》---------关键字(1)

1.双击实质--->加载内存 windows系统里面&#xff0c;双击的本质就是运行程序&#xff0c;把程序加载到内存里面&#xff1b; 任何程序运行的时候都必须加载到内存里面&#xff1b; 程序没有运行之前在硬盘里面&#xff0c;为什么程序运行之前必须加载到内存里面呢&#…

【Apache ShenYu源码】如何实现负载均衡模块设计

ShenYu是一个异步的&#xff0c;高性能的&#xff0c;跨语言的&#xff0c;响应式的 API 网关。有关ShenYu的介绍可以戳这。 一、前瞻 今天我们尝试不同的代码阅读方式&#xff0c;按模块来去阅读源码&#xff0c;看看效果如何。 本次阅读锁定在shenyu-loadbalancer&#xf…

比较基因组——还是看我的教程吧!

一、运行orthofinder 首先 orthofinder使用的版本为2.5.* 不要使用2.2的&#xff0c;2.2默认比对是blast&#xff0c;速度非常慢&#xff0c;结果文件呈现形式也不让人满意。2.5默认用的diamond 速度非常快 第一步代码&#xff1a; nohup orthofinder -t 40 -f data/ & # …

android 的 hal-jni

目标是写一个APP调用HAL的一个函数,在AOSP源码环境下进行开发,大概流程是: APP---->Framework service---->native----->HAL 1.什么是HAL HAL全称Hardware Abstract Layer,硬件抽象层,它向下屏蔽了硬件的实现细节,向上提供了抽象接口,HAL是底层硬件和上层框架…

【网页实战项目设计】基于SSM的医院预约挂号系统

基于SSM的医院预约挂号系统 项目截图 开发环境与技术框架 开发语言&#xff1a;Java 框架&#xff1a;ssm 技术&#xff1a;JSP JDK版本&#xff1a;JDK1.8 服务器&#xff1a;tomcat7 数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09; 数据库工具&a…

实战whisper第二天:直播语音转字幕(全部代码和详细部署步骤)

直播语音实时转字幕&#xff1a; 基于Whisper的实时直播语音转录或翻译是一项使用OpenAI的Whisper模型实现的技术&#xff0c;它能够实时将直播中的语音内容转录成文本&#xff0c;甚至翻译成另一种语言。这一过程大致分为三个步骤&#xff1a;捕获直播音频流、语音识别&#x…

面试算法-70-合并 K 个升序链表

题目 给你一个链表数组&#xff0c;每个链表都已经按升序排列。 请你将所有链表合并到一个升序链表中&#xff0c;返回合并后的链表。 示例 1&#xff1a; 输入&#xff1a;lists [[1,4,5],[1,3,4],[2,6]] 输出&#xff1a;[1,1,2,3,4,4,5,6] 解释&#xff1a;链表数组如下…

深入了解RSA算法:公钥密码学的基石

RSA算法是一种非对称加密算法&#xff0c;由Ron Rivest、Adi Shamir和Leonard Adleman于1978年提出。它基于大数分解的困难性&#xff0c;使用一对公钥和私钥进行加密和解密。本文将详细介绍RSA算法的原理、实现以及应用场景。 一、RSA算法原理 大数分解问题 RSA算法的基础是…

在线教育话术(1W字精选)

产品结构图 Nginx实现代理 问&#xff1a;我们在本机的host文件中配置了域名映射&#xff0c;都是同一个服务器。我们只需要输入对应的域名就可以到对应的界面&#xff0c;这是怎么实现的&#xff1f; 答&#xff1a;主要就是通过Nginx反向代理来实现的&#xff0c;Nginx会先…

低代码开发与物联网应用:重塑未来技术生态

随着科技的飞速发展&#xff0c;物联网(IoT)正逐渐成为我们日常生活的重要组成部分。在这个大背景下&#xff0c;低代码开发平台的出现为物联网应用的开发带来了革命性的变化。本文旨在探讨低代码开发与物联网应用之间的关系&#xff0c;分析低代码开发如何推动物联网应用的快速…

Java学习笔记之Java基础语法02-运算符

1.运算符和表达式 1.1 运算符&#xff1a; 定义&#xff1a;就是对常量&#xff08;字面量&#xff09;或者变量进行操作的符号。 比如&#xff1a; - * / 1.2 表达式&#xff1a; 定义&#xff1a;用运算符把常量或者变量连接起来的&#xff0c;符合Java语法的式子就是表达…

2024-03-20 作业

作业要求&#xff1a; 1> 创建一个工人信息库&#xff0c;包含工号&#xff08;主键&#xff09;、姓名、年龄、薪资。 2> 添加三条工人信息&#xff08;可以完整信息&#xff0c;也可以非完整信息&#xff09; 3> 修改某一个工人的薪资&#xff08;确定的一个&#x…

电影aac是什么意思?如何播放、转换、编辑aac?

"电影AAC"这个术语可能是指电影中的音频编码格式。AAC&#xff08;Advanced Audio Coding&#xff09;是一种常见的音频编码格式&#xff0c;通常用于压缩音频文件&#xff0c;以在保持高质量的同时减小文件大小。在电影中&#xff0c;AAC格式的音频通常用于提供高质…

Java学习笔记NO.25

T2.编写程序实现乐手弹奏乐器。乐手可以弹奏不同的乐器从而发出不同的声音。可以弹奏的乐器包括二胡、钢琴和琵琶。要求&#xff1a; (1)定义乐器类Instrument&#xff0c;包括方法makeSound() (2)定义乐器类的子类&#xff1a;二胡Erhu、钢琴Piano和小提琴Violin (3)定义乐手类…

H12-811题库(带解析,亲测高分可以通过)

大家可以直接点赞关注后&#xff0c;加作者微信&#xff08;备注“CSDN”&#xff09;就可以获取&#xff0c;微信在文章最后&#xff01; 808、[单选题]某公司网管要进行网络规划的时候&#xff0c;能够要让PC1访问PC2的数据包从G0/0/0口走(图上G0/0/2)。PC2访问PC1的数据包从…

抗辐射相机行业深度分析及投资机会探讨

一、市场趋势 抗辐射相机作为特种设备&#xff0c;主要用于军事侦察、核工业检测、太空探索等高辐射环境下的成像需求。随着全球核能和太空探索活动的增加&#xff0c;对高性能抗辐射相机的需求呈现上升趋势。此外&#xff0c;国家安全防护级别的提升也为该行业带来了新的增长…

浅谈RPC的理解

浅谈RPC的理解 前言RPC体系Dubbo架构最后 前言 本文中部分知识涉及Dubbo&#xff0c;需要对Dubbo有一定的理解&#xff0c;且对源码有一定了解 如果不了解&#xff0c;可以参考学习我之前的文章&#xff1a; 浅谈Spring整合Dubbo源码&#xff08;Service和Reference注解部分&am…

网络世界的城关——网卡

网络世界的城关——网卡 网卡到底是什么&#xff1f;网卡的功能网卡的真面目网卡的组成网卡的种类1.基于网络连接方式分类2.基于总线接口类型分类3.基于接口类型的分类4.基于传输速度的分类5.基于应用领域的分类 网卡到底是什么&#xff1f; 网卡我们可以这样通俗地理解&#x…

vue 若依 新开tab 不关闭旧的tab

注意 这里不是新开窗口&#xff08;_blank不符合我的需求&#xff09;&#xff0c;是新开一个tab若依对于统一组件打开的tab只能存在一个&#xff0c;因为路径相同 思路 路径不一致不就行了&#xff0c;动态生成的法子肯定是用不了了在router.js中配置路径&#xff1a;路径后面…