当前位置: 首页 > news >正文

400行程序写一个实时操作系统(十):用面向对象思想构建抢占式内核

前言

通过前几章的学习,我们学会了如何为RTOS设计一个合理的内存管理算法。现在,是时候学习设计RTOS内核了。
关于RTOS内核的文章也有很多,但都有一点先射箭再化靶子的意味。要么是代码连篇解释却寥寥无几,要么是要先怎么样再怎么样的说教式教程。并不是说这样的教程不好,而是他们缺乏读者普遍需要的东西,也就是更关键的思想方法!

为此,笔者决定:从我们的需求与应用出发,通过从结果思考过程的方式,使用面向对象的思想,逐步构建一个RTOS的内核。

程序 = 数据结构 + 算法

至于为什么要使用面向对象的思想,是因为面向对象思想本身和程序 = 数据结构 + 算法思想就是相通的,我们可以通过类和对象来表现数据结构,通过方法实现算法,从对象与对象的交互关系来构建,从而实现更加健壮的程序。

另外再借用linus的一句话:“Bad programmers worry about the code. Good programmers worry about data structures and their relationships.”

我们需要实现什么?

如果读者有过使用RTOS的经历,那么请你思考:RTOS实现了什么?带来了怎样的便利?

笔者先提出一点:多线程与优先级带来的实时性

RTOS将应用程序划分为多个独立的任务,也就是多线程。多线程允许同时执行多个任务,提高系统的处理能力和效率。例如,在嵌入式系统中,一个线程可以处理传感器数据,另一个线程可以更新用户界面。

实时性的需求,要求RTOS必须在指定的时间内完成关键任务。

我们创建一个又一个的任务,知道这是一个又一个的线程,它们可以并行执行。设置优先级时,我们知道优先级高的任务会优先执行,从而满足实时性。当我们想让任务同时且更有主次地执行时,我们第一个想到的就是使用RTOS。那么,强大的实时性与同时执行的任务,这就是我们想要实现的结果!但是,我们该如何去实现它呢?

如何实现实时性?

请读者想一想,我们创建任务设置优先级时,往往希望某些任务被优先执行,这是实时性实现的关键,也就是说,会有一个调度器来选择高优先级的任务,因此,我们得到了两个对象:任务和调度器

优先级依次递减
调度器_选择当前优先级最高的任务执行
任务1
就绪列表存放任务
任务2
任务3
任务4

调度器对象

通过上图我们可以推断,调度器会选择就绪列表中优先级高的任务。同时,就绪列表经常会发生变化,当优先级最高的任务发生变化,那么调度器还要切换任务,因此,我们得到了下图:

调度器
初始化
选择优先级最高的任务启动
切换任务
切换任务

切换任务时,我们肯定不希望先前任务的状态丢失,因此需要保存任务状态。线程(任务)切换如下:

​ 1.保存之前运行的线程的上下文

​ 2.选择优先级高的任务

3.调用准备运行的线程的上下文

因此有了下图:

调度器
初始化
启动
切换任务
保存任务状态
选择优先级最高的任务
切换到下一个任务

保存任务状态,这部分就涉及到和任务对象的交互了。

任务对象

为了方便管理任务,比如设置优先级啥的,我们肯定需要一个任务控制块,也方便我们把任务挂载到就绪列表中。同时,我们要保存当前状态,也就是说我们需要内存,那么这段内存我们给它命名为栈。

任务
任务控制块
任务栈
任务控制块

一个任务需要记录任务栈的信息,也就是pxTopOfStack(栈顶)、pxStack(栈起始地址)、self_stack(栈对象)这三个结构体。为了实时性,我们还需要优先级。

把图进一步展开:

任务
任务控制块
任务栈
栈顶
栈起始地址
栈对象
优先级

现在,我们关键的数据结构已经出来了,请读者写下这些代码,pxCurrentTCB就是当前执行的优先级最高的任务了:

sparrow.cClass(TCB_t)
{volatile uint32_t * pxTopOfStack;unsigned long uxPriority;uint32_t * pxStack;Stack_register *self_stack;
};
typedef  TCB_t         *TaskHandle_t;__attribute__( ( used ) )  TCB_t * volatile pxCurrentTCB = NULL;
typedef void (* TaskFunction_t)( void * );
栈对象

对于栈对象,我们要保存先前的任务状态,方便下一次任务执行时取出当前任务状态到CPU中,那么任务状态是CPU中的哪些信息呢?答案是寄存器:

img

以及XPSR,它是非常重要的特殊寄存器:

img

寄存器包括两部分寄存器,一部分是发生中断时硬件自动帮我们保存的寄存器,另一部分是需要我们手动保存的寄存器。

因此继续展开我们的图:

任务
任务控制块
任务栈
寄存器
自动保存的寄存器
手动保存的寄存器
栈顶
栈起始地址
栈对象
优先级

因此让我们写下代码:

Class(Stack_register)
{//automatic stackinguint32_t r4;uint32_t r5;uint32_t r6;uint32_t r7;uint32_t r8;uint32_t r9;uint32_t r10;uint32_t r11;//manual stackinguint32_t r0;uint32_t r1;uint32_t r2;uint32_t r3;uint32_t r12;uint32_t LR;uint32_t PC;uint32_t xPSR;
};

总结

现在,我们的大蓝图已经构建完毕,各种对象与它们之间的关系已然跃出水面。准备就绪,是时候去实现一个RTOS了!

相关文章:

400行程序写一个实时操作系统(十):用面向对象思想构建抢占式内核

前言 通过前几章的学习,我们学会了如何为RTOS设计一个合理的内存管理算法。现在,是时候学习设计RTOS内核了。 关于RTOS内核的文章也有很多,但都有一点先射箭再化靶子的意味。要么是代码连篇解释却寥寥无几,要么是要先怎么样再怎么…...

C#学习笔记(九)

C#学习笔记(九) 第六章 面向对象编程(一)类与对象、字段与属性一、类与对象正确的理解1. 什么是类?2.什么是对象?3. 类与对象的区别 二、类的基本规范和对象使用1. 类的规范 三、类的访问修饰符&#xff08…...

意外发现!AI写作这样用,热点文章轻松超越同行90%!

做自媒体,写热点文章很重要。 热点自带流量,能很快吸引不少读者。 可很多自媒体新手很犯愁。 干货文还能勉强写出来,碰到热点文就不知咋办了。 为啥写热点文章这么难呢? 关键是得找个新颖角度切入。 要是只在网上反复复制粘贴那些…...

WPF常见容器全方位介绍

Windows Presentation Foundation (WPF) 是微软的一种用于构建Windows桌面应用程序的UI框架。WPF的布局系统基于容器,帮助开发者以灵活、响应的方式组织用户界面 (UI) 元素。本篇文章将详细介绍WPF中几种常见的容器,包括Grid、StackPanel、WrapPanel、Do…...

重置时把el-tree树节点选中状态取消

要重置 Element UI 的 el-tree 组件并取消所有节点的选中状态,可以通过以下几种方法: 使用 setCheckedKeys 方法: 如果你的树配置了 node-key 属性,可以使用 setCheckedKeys 方法来清空所有选中的节点。 this.$refs.tree.setCheck…...

服务器系统克隆技术

工作任务:克隆对象是Windows server2019 和2022的datacenter版本 条件:在已经完成安装的虚拟机上做克隆 图1-1 用两个服务器的母盘准备进行克隆 第一步:新建一个文件目录用于安放克隆好的服务器 图1-2 创建两个目录用于安放即将克隆好的服务…...

【Java】多线程 Start() 与 run() (简洁实操)

Java系列文章目录 补充内容 Windows通过SSH连接Linux 第一章 Linux基本命令的学习与Linux历史 文章目录 Java系列文章目录一、前言二、学习内容:三、问题描述start() 方法run() 方法 四、解决方案:4.1 重复调用 .run()4.2 重复调用 start()4.3 正常调用…...

基于微信小程序的购物系统【附源码、文档】

博主介绍:✌IT徐师兄、7年大厂程序员经历。全网粉丝15W、csdn博客专家、掘金/华为云//InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ 🍅文末获取源码联系🍅 👇🏻 精彩专栏推荐订阅👇&#x1f3…...

AI绘画:24最新Stable Diffusion 终极炼丹宝典:从入门到精通!

前言 我是咪咪酱,以浅显易懂的方式,与大家分享那些实实在在可行之宝藏。 历经耗时数十个小时,总算将这份Stable Diffusion的使用教程整理妥当。 从最初的安装与配置,细至界面功能的详解,再至实战案例的制作&#xf…...

线性可分支持向量机的原理推导【补充知识部分】拉格朗日函数 公式解析

本文是将文章《线性可分支持向量机的原理推导》中的公式单独拿出来做一个详细的解析,便于初学者更好的理解。在主文章中,有一个部分是关于补充拉格朗日对偶性的相关知识,此公式即为这部分内容。 公式 9-9 是关于拉格朗日函数 L ( x , α , β…...

csdn(最新交流群)

SEOI Chathttps://seoi.net/room/10122?kwe7cp45v此网站开放性较强,小心诈骗...

新手maven入门学习教程

MAVEN基础入门 提示:java新人的学习之路记录 学习内容: 提示:了解并会初步使用maven构建管理java项目 Maven 是一个非常流行的 Java 项目管理和构建工具。它通过提供一套标准的构建生命周期和一组预定义的目标来简化 Java 应用程序的构建过…...

React 中级阶段学习计划

React 中级阶段学习计划 目标 掌握状态管理和路由。能够调用API并处理异步数据。学会使用CSS-in-JS和CSS Modules进行样式处理。 学习内容 状态管理 React Context API Context API:用于在组件树中传递数据,避免多层props传递。示例:im…...

[产品管理-47]:产品市场调研 - 一级市场、二级市场、次级市场?

目录 一、产品销售环节的一级二级市场 1、一级市场 2、二级市场 3、一级市场与二级市场的互动关系 二、金融中的一级二级市场 1、一级市场(Primary Market)- 新股发行、定向发行 2、二级市场(Secondary Market)- 普通投资者…...

Linux零基础教程学习(黑马)

1.初识Linux 1.2远程连接Linux系统 图形化、命令行 对于操作系统的使用,有2种使用形式: 图形化页面使用操作系统 以命令的形式使用操作系统 不论是Windows还是Linux亦或是MacOS系统,都是支持这两种使用形式。 图形化:使用操作…...

一款零依赖、跨平台的流媒体协议处理工具,支持 RTSP、WebRTC、RTMP 等视频流协议的处理

大家好,今天给大家分享一款功能强大的流媒体协议处理工具go2rtc,支持多种协议和操作系统,具有零依赖、零配置、低延迟等特点。 项目介绍 go2rtc可以从各种来源获取流,包括 RTSP、WebRTC、HomeKit、FFmpeg、RTMP 等,并…...

PHP 正则验证A-Z且排除某字母

都已经找到这里来了,相信已经尝试很多办法了,那么我们直接上答案 关键正则:(?!.*[IO]) //验证5到6个大写字母且排除I和O if (preg_match(/^(?!.*[IO])[A-Z\d]{5,6}$/u, AAAAM)) {echo "匹配成功"; } else {echo "匹配失败…...

如何安全运行别人上传的Python代码?

写后端的同学,有时候需要在网站上实现一个功能,让用户上传或者编写自己的Python代码。后端再运行这些代码。 涉及到用户自己上传代码,我们第一个想到的问题,就是如何避免用户编写危险命令。如果用户的代码里面涉及到下面两行&…...

matlab相位图

% 清空工作空间和命令窗口 clear; clc; % 模拟生成时间t,位移y(t)和角位移theta(t) t linspace(0, 100, 1000); % 时间从0到100,包含1000个点 y 1e-5 * sin(2 * pi * 0.1 * t) .* exp(-0.01 * t); % 位移y(t) 振荡衰减 theta 1e-6 * cos(2 * pi * …...

C语言笔记(指针的进阶)

目录 1.字符指针 2.指针数组 3.数组指针 3.1.创建数组指针 3.2.&数组名和数组名 1.字符指针 int main() { char ch w;char* pc &ch;const char *p "abcdef";//常量字符串 产生的值就是首元素的地址//常量字符串不能被修改 因此需要加上一个…...

OpenLayers 可视化之热力图

注:当前使用的是 ol 5.3.0 版本,天地图使用的key请到天地图官网申请,并替换为自己的key 热力图(Heatmap)又叫热点图,是一种通过特殊高亮显示事物密度分布、变化趋势的数据可视化技术。采用颜色的深浅来显示…...

使用VSCode开发Django指南

使用VSCode开发Django指南 一、概述 Django 是一个高级 Python 框架,专为快速、安全和可扩展的 Web 开发而设计。Django 包含对 URL 路由、页面模板和数据处理的丰富支持。 本文将创建一个简单的 Django 应用,其中包含三个使用通用基本模板的页面。在此…...

CMake 从 GitHub 下载第三方库并使用

有时我们希望直接使用 GitHub 上的开源库,而不想手动下载、编译和安装。 可以利用 CMake 提供的 FetchContent 模块来实现自动下载、构建和链接第三方库。 FetchContent 命令官方文档✅ 示例代码 我们将以 fmt 这个流行的格式化库为例,演示如何: 使用 FetchContent 从 GitH…...

JVM暂停(Stop-The-World,STW)的原因分类及对应排查方案

JVM暂停(Stop-The-World,STW)的完整原因分类及对应排查方案,结合JVM运行机制和常见故障场景整理而成: 一、GC相关暂停​​ 1. ​​安全点(Safepoint)阻塞​​ ​​现象​​:JVM暂停但无GC日志,日志显示No GCs detected。​​原因​​:JVM等待所有线程进入安全点(如…...

3-11单元格区域边界定位(End属性)学习笔记

返回一个Range 对象,只读。该对象代表包含源区域的区域上端下端左端右端的最后一个单元格。等同于按键 End 向上键(End(xlUp))、End向下键(End(xlDown))、End向左键(End(xlToLeft)End向右键(End(xlToRight)) 注意:它移动的位置必须是相连的有内容的单元格…...

微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据

微软PowerBI考试 PL300-在 Power BI 中清理、转换和加载数据 Power Query 具有大量专门帮助您清理和准备数据以供分析的功能。 您将了解如何简化复杂模型、更改数据类型、重命名对象和透视数据。 您还将了解如何分析列,以便知晓哪些列包含有价值的数据,…...

人机融合智能 | “人智交互”跨学科新领域

本文系统地提出基于“以人为中心AI(HCAI)”理念的人-人工智能交互(人智交互)这一跨学科新领域及框架,定义人智交互领域的理念、基本理论和关键问题、方法、开发流程和参与团队等,阐述提出人智交互新领域的意义。然后,提出人智交互研究的三种新范式取向以及它们的意义。最后,总结…...

基于IDIG-GAN的小样本电机轴承故障诊断

目录 🔍 核心问题 一、IDIG-GAN模型原理 1. 整体架构 2. 核心创新点 (1) ​梯度归一化(Gradient Normalization)​​ (2) ​判别器梯度间隙正则化(Discriminator Gradient Gap Regularization)​​ (3) ​自注意力机制(Self-Attention)​​ 3. 完整损失函数 二…...

Web中间件--tomcat学习

Web中间件–tomcat Java虚拟机详解 什么是JAVA虚拟机 Java虚拟机是一个抽象的计算机,它可以执行Java字节码。Java虚拟机是Java平台的一部分,Java平台由Java语言、Java API和Java虚拟机组成。Java虚拟机的主要作用是将Java字节码转换为机器代码&#x…...

跨平台商品数据接口的标准化与规范化发展路径:淘宝京东拼多多的最新实践

在电商行业蓬勃发展的当下,多平台运营已成为众多商家的必然选择。然而,不同电商平台在商品数据接口方面存在差异,导致商家在跨平台运营时面临诸多挑战,如数据对接困难、运营效率低下、用户体验不一致等。跨平台商品数据接口的标准…...