手写嵌入式操作系统(基于stm8单片机)

#include <stc8h.h>
#include <intrins.h>
#define MAX_TASKS 2 //简化方面,我们当前操作系统只有2个task
#define MAX_TASK_DEP 32unsigned char idata task_sp[MAX_TASKS];  // 任务的堆栈指针
unsigned char idata task_stack[MAX_TASKS][MAX_TASK_DEP];// 每个一个task任务的堆栈
unsigned char task_id; //当前任务号, 从0开始。//任务状态
typedef enum{TASK_RUNNING, //运行状态TASK_SUSPENDED  //挂起状态
} TaskStatus;typedef struct{unsigned char id; //任务idTaskStatus status; //任务状态unsigned int delay_count; // 延迟计数器unsigned int delay_duration;//延迟时间
}Task ;Task idata tasks[MAX_TASKS] = {{0, TASK_RUNNING,0,0},{1, TASK_RUNNING,0,0},   //两个任务,都是默认运行状态,不延时,
};void Timer0_init(void); //原型函数 
void sleep(unsigned int , unsigned int );// tid, delay_msvoid sleep(unsigned int task_id , unsigned int delay_ms){tasks[task_id].status = TASK_SUSPENDED;tasks[task_id].delay_count = 0;tasks[task_id].delay_duration = delay_ms;}void Timer0_init(){AUXR |= 0x80;		//定时器时钟1T模式TMOD &= 0xF0;		//设置定时器模式EA = 1; // 全局中断允许ET0 = 1; // 定时器0中断允许TR0 = 1;		//定时器0开始计时TL0 = 0x40;		//设置定时初始值TH0 = 0xA2;		//设置定时初始值}void Delay1000ms()		//@24.000MHz
{unsigned char i, j, k;_nop_();_nop_();i = 122;j = 193;k = 128;do{do{while (--k);} while (--j);} while (--i);
}//定义一个任务切换的函数(任务调度器)
void task_switch(){task_sp[task_id] = SP;// 把当前系统的堆栈指针存入到某个小朋友的task_sp里面。task_id = task_id + 1; //任务加1if(task_id == MAX_TASKS){	 task_id = 0;} SP = task_sp[task_id];
}void task0(){//第0号任务, 代表第0个小朋友做的事情。//static unsigned int a = 3;P5M0 = 0x00;P5M1 = 0x00;P53 = 1;while(1){//a = a + 3;//Delay1000ms();//检查自己的状态。如果自己是睡眠状态,就应该交给别的task去执行if(tasks[0].status == TASK_SUSPENDED){task_switch();continue;}sleep(0,1000);P53 = ~P53;task_switch();}
}void task1(){//第1号任务, 代表第1个小朋友做的事情。//static unsigned int b = 5;P4M1 = 0x00;P4M0 = 0x00;P2M1 = 0x00;P2M0 = 0x00;P27 = 0;while(1){//检查自己的状态。如果自己是睡眠状态,就应该交给别的task去执行if(tasks[1].status == TASK_SUSPENDED){task_switch();continue;}//b = b + 5;//Delay1000ms();sleep(1,1000);P27 = ~P27;	task_switch();}
}void Timer0_ISR(void) interrupt 1 {//系统的定时器中断, 每隔1毫秒就执行一下中断函数unsigned char i;for(i =0 ;i<MAX_TASKS;i++){if(tasks[i].status == TASK_SUSPENDED){tasks[i].delay_count++;}if(tasks[i].delay_count >= tasks[i].delay_duration){tasks[i].status = TASK_RUNNING; //睡眠结束		 tasks[i].delay_count = 0;}}}void Timer1_Init(void)		//100微秒@24.000MHz
{AUXR |= 0x40;			//定时器时钟1T模式TMOD &= 0x0F;			//设置定时器模式TL1 = 0xA0;				//设置定时初始值TH1 = 0xF6;				//设置定时初始值TF1 = 0;				//清除TF1标志TR1 = 1;				//定时器1开始计时
}void Timer1_ISR(void) interrupt 3{task_switch();// 在timer1中的中断,进行任务切换。
}//幼儿园老师(操作系统,加载任务的函数)
//fn fn是一个函数的指针,注意数据类型是int 16位的。
//tid task id, 是8位的, 0,1 
//下面函数的作用就是把一个task的函数指针放入对应的堆栈空间里面。
void task_load(unsigned int fn, unsigned char tid){task_sp[tid] =  task_stack[tid] + 1; // 把任务的指针往下一个空间挪一格,两个char了task_stack[tid][0] = fn& 0xff;task_stack[tid][1] = fn>>8;	
}void main(){Timer0_init();Timer1_Init();task_load(task0,0);// 把task0 装载到内存中。task_load(task1,1);// 把task1 装载到内存中。task_id = 0;   SP = task_sp[0];
}

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

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

相关文章

2023年7月京东彩妆市场品牌销售排行榜(京东数据挖掘)

鲸参谋监测的京东平台7月份彩妆市场销售数据已出炉&#xff01; 鲸参谋数据显示&#xff0c;7月份彩妆市场整体呈现下滑趋势。从大盘数据可知&#xff0c;京东7月的销量将近350万&#xff0c;环比下滑约38%&#xff0c;同比下滑约22%&#xff1b;销售额为5.1亿&#xff0c;环比…

Integer、Long 等包装类 == 值判断、地址判断与缓存

先看下以下代码和输出 public static void main(String[] args) throws Exception{Integer a-128;Integer aa-128;System.out.printf("aaa? %s \n",aaa);Integer b127;Integer bb127;System.out.printf("bbb? %s \n",bbb);Integer c128;Integer cc128;Sy…

stable diffusion实践操作-LLuL 插件-局部修改

系列文章目录 大家移步下面链接中&#xff0c;里面详细介绍了stable diffusion的原理&#xff0c;操作等&#xff08;本文只是下面系列文章的一个写作模板&#xff09;。 stable diffusion实践操作 提示&#xff1a;写完文章后&#xff0c;目录可以自动生成&#xff0c;如何生…

代码随想录day22

235. 二叉搜索树的最近公共祖先 ● 力扣题目链接 ● 给定一个二叉搜索树, 找到该树中两个指定节点的最近公共祖先。 思路 ● 和普通二叉树相比&#xff0c;简单很多 ● 如果落在了[p, q]之间&#xff0c;返回root&#xff1b;都小就向右看&#xff0c;都大就向左看 代码 c…

JVM垃圾回收机制和常用算法(简洁版)

垃圾收集 (Garbage Collection,GC) 垃圾收集主要是针对堆和方法区进行。程序计数器、虚拟机栈和本地方法栈这三个区域属于线程私有的&#xff0c;只存在于线程的生命周期内&#xff0c;线程结束之后就会消失&#xff0c;因此不需要对这三个区域进行垃圾回收。 判断一个对象是…

浅析Keil MDK下串行Flash的下载算法设计

浅析Keil MDK下串行Flash的下载算法设计-电子发烧友网 今天给大家介绍的是 Keil MDK 工具下 i.MXRT 的串行 NOR Flash 下载算法设计。 在 i.MXRT 硬件那些事系列之《在串行 NOR Flash XIP 调试原理》一文中&#xff0c;痞子衡简单提了一下串行 NOR Flash 下载算法的概念&…

使用词袋模型(BoW)测试提取图像的特征点和聚类中心

文章目录 环境配置代码测试 环境配置 (1) 导入opencv&#xff0c;参考链接 https://blog.csdn.net/Aer_7z/article/details/132612369(2) 安装numpy 激活虚拟环境的前提下&#xff0c;输入&#xff1a; pip install numpy(3) 安装sklearn 激活虚拟环境的前提下&#xff0c;输…

数据结构前言

一、什么是数据结构&#xff1f; 数据结构是计算机存储、组织数据的方式。数据结构是指相互之间存在一种或多种特定关系的数据元素的集合。 上面是百度百科的定义&#xff0c;通俗的来讲数据结构就是数据元素集合与数据元素集合或者数据元素与数据元素之间的组成形式。 举个…

vs中git提交合并分支的步骤记录

vs打开终端 PS D:\project\et_lower4_driver> git pull Already up to date. PS D:\project\et_lower4_driver> git branch * kiyun_usb7851 master PS D:\project\et_lower4_driver> git checkout master Switched to branch master Your branch is up to date wit…

MySQL与postgreSQL数据库的区别

MySQL 是一个流行的开源关系型数据库管理系统&#xff0c;具有以下优势&#xff1a; 开源和免费&#xff1a;MySQL 是一个开源软件&#xff0c;允许用户免费下载、使用和修改。它的免费版本&#xff08;Community Edition&#xff09;提供了广泛的功能&#xff0c;适用于大多数…

操作系统(OS)与系统进程

操作系统&#xff08;OS&#xff09;与系统进程 冯诺依曼体系结构操作系统(Operator System)进程基本概念进程的描述&#xff08;PCB&#xff09;查看进程通过系统调用获取进程标示符&#xff08;PID&#xff09;通过系统调用创建进程&#xff08;fork&#xff09;进程状态&…

小程序进阶-env(safe-area-inset-bottom)的使用

一、简介 env(safe-area-inset-bottom)和env(safe-area-inset-top)是CSS中的变量&#xff0c;用于获取设备底部和顶部安全区域的大小。 所谓的安全区域就是指在iPhone X及以上的设备中&#xff0c;为避免被屏幕的“刘海”和“Home Indicator”所遮挡或者覆盖的有效区域区域&am…

UICollectioView 使用集合

文章目录 一、前言二、UICollectionView基础知识2.1多个对象协作而成2.2 DataSource与Delegate三、UICollectionView 常用操作3.1 允许cell多选、 设置默认选中、选中与取消选中控制3.2 显示编辑按钮3.3 layouts 布局之间的转场3.4 插入、删除、移动 sections 和 items3.5 更新…

解决解析maven依赖加载卡住问题

首先找到用户设置文件的位置&#xff1a; 根据位置查找文件&#xff0c;没有则创建。 修改maven的默认镜像为阿里云镜像。 <mirror><id>alimaven</id><name>aliyun maven</name><url>http://maven.aliyun.com/nexus/content/groups/pub…

实训笔记9.4

实训笔记9.4 git一、本地版本控制1.1 建立仓库1.2 添加到暂存区1.3 提交到历史区1.4 查看1.4.1 查看当前版本库状态1.4.2 查看版本日志信息1.4.3 查看版本的变更历史 1.5 撤销1.5.1 撤销工作区的修改1.5.2 撤销暂存1.5.3 历史穿梭 二、代码托管2.1 将本地仓库推送到远程平台2.2…

【C++】STL-函数对象 + 谓词

1.函数对象使用 #include <iostream> using namespace std;//STL-函数对象&#xff08;仿函数&#xff09;class MyAdd { public:int operator()(int v1, int v2){return v1 v2;} }; //1、函数对象在使用时&#xff0c;可以像普通函数那用调用&#xff0c;可以有参数&am…

视频汇聚/视频云存储/视频监控管理平台EasyCVR接入海康SDK协议后无法播放该如何解决?

开源EasyDarwin视频监控/安防监控/视频汇聚EasyCVR能在复杂的网络环境中&#xff0c;将分散的各类视频资源进行统一汇聚、整合、集中管理&#xff0c;在视频监控播放上&#xff0c;视频安防监控汇聚平台可支持1、4、9、16个画面窗口播放&#xff0c;可同时播放多路视频流&#…

vue 设置全局鼠标移动事件

要设置全局鼠标移动事件&#xff0c;可以使用 Vue 的 mixin 实现。在 mixin 中&#xff0c;我们可以使用 $on 方法监听 mousemove 事件&#xff0c;并在组件销毁时使用 $off 方法移除监听器&#xff0c;以避免内存泄漏。以下是一个例子&#xff1a; // 在全局中注册一个 mixin…

基于nRF52840 Dongle配合Wireshark对Mesh网络抓包并解析(Nordic)

Mesh网络抓包解析 准备说明配置过滤解密分析 准备 1&#xff09;nRF52840 Dongle 2&#xff09;Mesh节点 3&#xff09;手机作为配网器&#xff08;苹果手机安装nRF Mesh APP&#xff09; 说明 1&#xff09;节点使用的例程目录&#xff1a;nrf5sdkformeshv500src\examples\…

java八股文面试[多线程]——阻塞队列

阻塞队列大纲&#xff1a; 什么是阻塞队列 阻塞队列&#xff1a;从名字可以看出&#xff0c;他也是队列的一种&#xff0c;那么他肯定是一个先进先出&#xff08;FIFO&#xff09;的数据结构。与普通队列不同的是&#xff0c;他支持两个附加操作&#xff0c;即阻塞添加和阻塞删…