从零手写操作系统之RVOS硬件定时器-05

news2024/12/23 10:25:36

从零手写操作系统之RVOS硬件定时器-05

  • RISC-V 定时器中断
  • RISC-V CLINT 介绍
    • 寄存器 (Timer 部分)
    • 总体框架流程
    • 硬件定时器的应用
      • 时间管理
      • 测试


本系列参考: 学习开发一个RISC-V上的操作系统 - 汪辰 - 2021春 整理而来,主要作为xv6操作系统学习的一个前置基础。

RVOS是本课程基于RISC-V搭建的简易操作系统名称。

课程代码和环境搭建教程参考github仓库: https://github.com/plctlab/riscv-operating-system-mooc/blob/main/howto-run-with-ubuntu1804_zh.md

前置知识:

  • RVOS环境搭建-01
  • RVOS操作系统内存管理简单实现-02
  • RVOS操作系统协作式多任务切换实现-03
  • RISC-V 学习篇之特权架构下的中断异常处理
  • 从零手写操作系统之RVOS外设中断实现-04

RISC-V 定时器中断

定时器中断属于本地中断的一类:
在这里插入图片描述

Core Local INTerrupt:

在这里插入图片描述
Core Local Interrupt(CLINT)是一个与处理器核心相关的中断控制器,它负责处理特定核心的计时器中断和软件中断。

CLINT位于RISC-V系统中的物理内存地址空间,它是一个全局共享的设备,被所有的处理器核心共享和访问。CLINT的作用是为每个处理器核心提供计时器中断和软件中断的控制。

CLINT通常具有以下功能和组成部分:

  1. Timer Interrupts(计时器中断):CLINT包含一个或多个计时器,用于生成定时中断。每个计时器都与特定的处理器核心关联,当计时器计数达到预设的值时,CLINT会生成一个中断信号,通知相应的处理器核心。

  2. Software Interrupts(软件中断):CLINT可以接收来自处理器核心的软件中断请求。软件中断是由软件代码触发的一种中断请求,用于实现系统调用、任务切换和异常处理等功能。

  3. 中断控制寄存器(Interrupt Control Registers):CLINT包含一组用于配置和控制中断的寄存器,包括计时器设置寄存器、中断使能寄存器、中断优先级寄存器等。这些寄存器用于配置中断参数、使能或禁用中断,并设置中断的优先级。

总而言之,CLINT是一个处理器核心本地的中断控制器,它提供定时器中断和软件中断的功能,并通过相关的寄存器进行配置和控制。每个处理器核心都可以访问和配置CLINT,以实现对中断的管理和处理。


RISC-V CLINT 介绍

在这里插入图片描述

寄存器 (Timer 部分)

在这里插入图片描述

全局唯一,表示即使存在多个核,也只会存在一个mtime寄存器

mtime是RISC-V架构中的一个特殊寄存器,用于表示机器模式下的计时器值。它是Machine Timer(机器计时器)的缩写。

mtime寄存器通常由硬件提供,用于跟踪系统运行的时间。它的值会不断增加,可以用于测量程序的执行时间、进行时间相关的操作和调度等。

在RISC-V中,mtime寄存器是一个64位的寄存器,可用于测量长时间间隔,通常以时钟周期或计时器滴答数的形式表示。它的精度和计时精度取决于硬件实现和操作系统的支持。

在操作系统或应用程序中,可以使用mtime寄存器来实现计时器、延时函数、性能统计等功能。通过读取mtime寄存器的值,可以获得当前的计时器数值,进而进行时间计算和处理。

需要注意的是,访问mtime寄存器通常需要特权级别的权限。在特权级别较低的用户态,可能无法直接读取或写入mtime寄存器,需要通过系统调用或特权级别切换来访问。具体的访问权限和操作方式取决于系统的实现和配置。

  • mtime寄存器的递增原理是由硬件实现确定的,通常是由时钟或计时器驱动的。

  • 在一个基于时钟的系统中,系统时钟会以固定的频率进行振荡,产生一个稳定的时钟信号。这个时钟信号会被用作各种硬件模块和功能的时序控制。

  • mtime寄存器会根据系统时钟信号的脉冲进行递增。每当一个时钟脉冲到达,mtime寄存器的值会自动加1。这样,随着时钟信号的不断变化,mtime寄存器的值也会不断地增加。

  • 递增速度取决于时钟的频率。如果系统时钟频率为1 MHz,那么每秒钟mtime寄存器的值就会增加1000000。因此,可以根据mtime寄存器的递增速度来进行时间计算和测量。

  • 需要注意的是,mtime寄存器的递增是硬件自动完成的,无法通过软件或程序直接控制。程序可以通过读取mtime寄存器的值来获取当前的计时器数值,但无法直接修改或控制其递增过程。递增过程是由硬件实现和时钟信号控制的,程序只能观察和利用其递增的结果。

在这里插入图片描述
mtimecmp寄存器是RISC-V架构中的一个定时器比较寄存器(Timer Compare Register)。它用于与mtime寄存器进行比较,以实现定时器中断的触发。

mtime寄存器的值与mtimecmp寄存器的值相等时,会触发一个定时器中断。这种机制允许程序根据需要设置定时器中断的触发时机。

具体而言,程序可以通过向mtimecmp寄存器写入一个比较值,来指定何时触发定时器中断。当mtime寄存器的值达到或超过这个比较值时,定时器中断被触发,执行相应的中断处理程序。

通过使用mtimecmp寄存器,程序可以实现定时器相关的功能,如定时任务调度、时间片轮转调度、精确延时等。它为程序提供了一种基于时间的触发机制,使得程序能够按照预定的时间间隔执行特定的操作。

需要注意的是,具体的定时器中断触发机制和中断处理程序的实现方式可能会有所不同,取决于具体的处理器和操作系统。程序需要根据所使用的平台和系统进行相应的配置和编程。

硬件定时器初始化代码如下:

/* 10000000 ticks per-second */
#define CLINT_TIMEBASE_FREQ 10000000

/* interval ~= 1s */
#define TIMER_INTERVAL CLINT_TIMEBASE_FREQ


void timer_init()
{
	/*
	 * On reset, mtime is cleared to zero, but the mtimecmp registers 
	 * are not reset. So we have to init the mtimecmp manually.
	 */
	//定时器模块初始化---传入interval间隔大约为1s
	timer_load(TIMER_INTERVAL);

	/* enable machine-mode timer interrupts. */
	//开启次级中断中的定时器中断
	w_mie(r_mie() | MIE_MTIE);

	/* enable machine-mode global interrupts. */
	//开启全局中断
	w_mstatus(r_mstatus() | MSTATUS_MIE);
}
/* load timer interval(in ticks) for next timer interrupt.*/
void timer_load(int interval)
{
	/* each CPU has a separate source of timer interrupts. */
	//获取当前hartId
	int id = r_mhartid();
	//设置mtimecmp寄存器的值为mtime寄存器的值+interval
	*(uint64_t*)CLINT_MTIMECMP(id) = *(uint64_t*)CLINT_MTIME + interval;
}

经过如上设置后,大约1秒后,会触发一次时钟中断。


在这里插入图片描述
mtime中断发生时,处理器核心(hart)会设置mip寄存器的MTIP位,表示发生了定时器中断。

在处理定时器中断时,通常需要在mtimecmp寄存器中写入新的值以清除mip.MTIP位。具体的操作步骤如下:

  1. 响应定时器中断,进入中断处理程序。
  2. 在中断处理程序中,读取mtime寄存器的当前值,可以使用类似于uint64_t curr_time = r_mtime();的方式获取。
  3. 根据需要,计算下一个定时器中断应该发生的时间,得到一个新的比较值。
  4. 将新的比较值写入mtimecmp寄存器,以设置下一个定时器中断的触发时刻。
  5. 清除mip寄存器的MTIP位,以告知处理器中断已经处理完毕。可以使用类似于w_mip(r_mip() & ~MIP_MTIP);的方式清除。

通过在中断处理程序中更新mtimecmp寄存器,程序可以实现周期性的定时器中断,不断触发指定时间间隔的操作。同时,清除mip.MTIP位可以确保处理器核心在中断处理程序执行完毕后正确地处理下一个定时器中断。

需要注意的是,具体的操作方式可能因处理器和操作系统的不同而有所差异。因此,以上描述仅为一般情况下的操作流程,具体的实现方式需要参考所使用的平台和系统的文档或相关编程接口。

我们需要在trap_handler处理函数中新增对定时器中断的处理:

reg_t trap_handler(reg_t epc, reg_t cause)
{
	reg_t return_pc = epc;
	reg_t cause_code = cause & 0xfff;
	
	if (cause & 0x80000000) {
		/* Asynchronous trap - interrupt */
		switch (cause_code) {
		case 3:
			uart_puts("software interruption!\n");
			break;
	    //新增对定时器中断的处理		
		case 7:
			uart_puts("timer interruption!\n");
			timer_handler();
			break;
		case 11:
			uart_puts("external interruption!\n");
			external_interrupt_handler();
			break;
		default:
			uart_puts("unknown async exception!\n");
			break;
		}
	} else {
		/* Synchronous trap - exception */
		printf("Sync exceptions!, code = %d\n", cause_code);
		panic("OOPS! What can I do!");
		//return_pc += 4;
	}

	return return_pc;
}

这里是引用

  • 定时器中断具体处理函数
void timer_handler() 
{ 
    //记录定时器中断触发次数
	_tick++;
	printf("tick: %d\n", _tick);
    //设置下一次定时器中断触发时机
	timer_load(TIMER_INTERVAL);
}

总体框架流程

在这里插入图片描述


硬件定时器的应用

时间管理

在这里插入图片描述
在这里插入图片描述

/* interval ~= 1s */
#define TIMER_INTERVAL CLINT_TIMEBASE_FREQ

static uint32_t _tick = 0;

测试

我们这里简单测试一下定时器中断运行效果:

void start_kernel(void)
{
	uart_init();
	uart_puts("Hello, RVOS!\n");

	page_init();

	trap_init();

	plic_init();
    //初始化定时器模块
	timer_init();

	sched_init();

	os_main();

	schedule();

	uart_puts("Would not go here!\n");
	while (1) {}; // stop here!
}

在这里插入图片描述

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

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

相关文章

vm安装Ubuntu以及Ubuntu设置中文

vm安装Ubuntu以及Ubuntu设置中文 Ubuntu 介绍 Ubuntu(友帮拓、优般图、乌班图)是一个以桌面应用为主的开源GNU/Linux 操作系统,Ubuntu 是基于GNU/Linux,支持x86、amd64(即x64)和ppc 架构,由全…

基于DICOM标准的医院PACS系统源码

一、基于 DICOM 的 PACS是医学图像归档与通信系统,是实现医学图像自动获取、显示、图像 后处理、传输、存储、查询、检索、写诊断报告、查看成像设备运行状态等功能复合型医学 图像管理系统。 二、PACS可以为医院其他系统提供医学图像,并能够形成图文并茂…

Ansible进阶1——大项目清单管理

文章目录 一、playbook引用主机方式1.1 引用清单主机1.2 引用主机别名1.3 使用通配符引用 二、动态清单三、管理多个清单四、forks和serial五、包含与导入5.1 导入playbook5.2 导入任务文件5.3 包含任务文件5.4 结合变量 一、playbook引用主机方式 清单文件的使用方式很多&…

UNIX网络编程卷一 学习笔记 第十八章 路由套接字

内核中的Unix路由表传统上一直使用ioctl函数访问,但没有ioctl函数请求能获取整个路由表,而netstat等程序通过读取内核的内存获取路由表内容。诸如gated等路由守护进程需要监视由内核收取的ICMP重定向消息(用于在IP网络中通知主机或路由器更优…

【网络】UDP网络服务器

代码的整体逻辑: UDP服务端:udpServer.cc(服务端的调用),udpServer.hpp(服务端的实现) UDP客户端:udpClient.cc(客户端的调用),udpClient.hpp(客户端的实现) 1.udp服务…

Flask中使用Flask-WTF验证JSON对象参数

Flask中使用Flask-WTF验证JSON对象参数 1 安装 Flask-WTF是对WTForms库的扩展,默认只对”form“标签下的表单进行验证。针对当前前后端分离的项目中,json传输参数需要进行二次封装才能使用。 # 安装Flask-WTF包 pip install Flask-WTF2 源代码 2.1 工…

一些忘了的东西(二)

Symbol出现的原因/作用 ①作为属性 避免属性冲突重复,就是使用它来表示唯一值; 问题是我们什么情况下 要保障属性一定不冲突重复 覆盖呢? 在vue里 有this.$ parent ,this.$ options ,this.$ set 这些,使用$命名开头就是想通过命名约定来减少…

BLIP使用教程

文章目录 准备测试示例一示例二: 结论源代码 原理篇: BLIP2-图像文本预训练论文解读 准备 如果无网络需提前下载相关模型 安装torch、transformers pip install torch trtransformers测试 测试blip基于图片生成文本描述能力(Caption&…

《嵌入式系统》知识总结8:寄存器编程方式操纵GPIO

方式1: 查手册找到相关寄存器的地址,自行编写代码,定义指针指向该地址,并根据需要进行寄存器读写。 举例: //方式1举例:自定义GPIOB_ODR寄存器 #define GPIOB_ODR (*(volatile unsigned int *)0x40010C0…

Vivado时序约束基础

今天这篇博客,笔者向大家简单介绍Xilinx FPGA中的Vivado时序约束基础知识,也为后续的学习打好铺垫。 Xilinx Design Constraints (XDC) 概述 • XDC 在本质上就是Tcl 语言,但其仅支持基本的Tcl 语法,如变量、列表和运算符等等&a…

Go 语言基础

文章目录 Go 语言基础1. 程序基础2. 数据类型3. 字符串与复合数据类型4. 函数、方法、接口和反射5. 并发编程(核心重点)6. 包和代码测试7. 综合实战案例框架部分探索深度 Go 语言基础 1. 程序基础 了解常量和遍历【const var 关键词】 : 初始化以及赋值…

微服务如何治理

微服务远程调用可能有如下问题: 注册中心宕机; 服务提供者B有节点宕机; 服务消费者A和注册中心之间的网络不通; 服务提供者B和注册中心之间的网络不通; 服务消费者A和服务提供者B之间的网络不通; 服务提供者…

Redux And Redux Toolkit

笔记介绍了react_redux和redux_toolkit react_redux只介绍了原理图,为理解redux_toolkit做铺垫。 本笔记是对一下课程做的输出,若大家有不理解的地方,可看完课程后,再借助课程理解笔记内容,同时也鼓励大家对自己的听…

【环境搭建】40系一些奇奇怪怪的环境问题

【设备信息】我的设备是4070ti,支持cuda12.0,但是目前用的还是11.7 1)fatal error: cusparse.h: No such file or directory 因为cuda版本和改名的原因,这个在cuda版本中比较有效的解决办法是: sudo apt search libcusparse得到…

matplotlib实操

matplotlib实操 问题1.分析离网用户的基本特征:包括但不限于地市、年龄、网龄、融合类型、套餐分布、用户价值等,年龄、网龄、用户价值(ARPU)、MOU、DOU;数据预处理处理异常值地市分布县级分布年龄分布网龄分布性别与年龄分布融合类型套餐分布用户价值(ARPU)MOU(每用…

vulnhub靶场之hackme2-DCHP

Burp联动xray 1.信息收集 探测存活主机,输入:netdiscover -r 192.168.239.0/24 ,发现192.168.239.179存活。 对目标主机192.168.239.179进行端口扫描,发现存活22、80端口。 在浏览器上输入:http://192.168.239.179&a…

axios解决跨域问题

Vue3中使用axios访问聚合的天气API,出现跨域问题,需要在前端进行一些配置: 首先是修改vue.config.js: const { defineConfig } require(vue/cli-service) module.exports defineConfig({transpileDependencies: true,devServe…

vue2.0基础

文章目录 VUEVue2.0vue特点事件处理键盘事件计算属性监听watch深度监视绑定class样式条件渲染列表渲染列表过滤列表排序Vue.set()的使用Vue检测数组的原理Vue监测原理总结指令生命周期Vue componentVue配置文件vue.config.JS其他:组件自定义事件组件自定义事件解绑全…

【Mysql数据库从0到1】-入门基础篇--用户与权限管理

【Mysql数据库从0到1】-入门基础篇--用户与权限管理 🔻一、Mysql 用户管理1.1 🍃 Mysql服务器登录1.2 🍃 用户创建1.3 🍃 用户修改1.4 🍃 用户删除1.5 🍃 用户密码修改1.6 🍃 用户密码管理 &…

深入浅出C语言—【函数】上

目录 1.函数的概念 2.C语言函数的分类 2.1 库函数 2.1.1 strcpy库函数举例学习方式 2.1.2 库函数扩展知识 2.2 自定义函数 2.2.1求两个整数中的较大值 3. 函数的参数 3.1 实际参数(实参) 3.2 形式参数(形参) 4. 函数的…