1、掌握在STM32上移植嵌入式实时操作系统uC/OS-II的方法
2、掌握在STM32上移植uC/OS-II的基本步骤
3、掌握uC/OS-II的文件OS_CPU.H编写
4、掌握uC/OS-II的文件OS_CPU_C.C编写
5、掌握uC/OS-II的文件OS_CPU_A.S的编写
uCOS也叫uC/OS-II,它是由美国人Jean .J.Labrosse在1992年编写的一个嵌入式多任务实时操作系统。它具有以下特点:
(1)微型化。嵌入式系统芯片内部存储器的容量通常不会很大,一般也不配置外存,加之电源的容量较小以及外部设备的多样性,因而不允许嵌入式操作系统占用较多的资源,所以在保证应用功能的前提下,嵌入式操作系统的规模越小越好
(2)可裁剪性。嵌入式操作系统的运行平台多种多样,其宿主对象更是五花八门,所以要求嵌入式操作系统中提供的各个功能模块可以让用户根据需要选择使用,即要求它具有良好的可裁剪性
(3)实时性。目前嵌入式系统广泛应用于生产过程控制、数据采集、传输通信等场合,这些应用的共同特点就是要求系统能快速响应事件,因此要求嵌入式操作系统要有较强的实时性
(4)高可靠性。嵌入式系统广泛地应用于军事武器、航空航天、交通运输等重要领域,所以要求嵌入式操作系统必须有极高的可靠性,对关键、要害的应用还要提供必要的容错和防错措施,以进一步提高系统的可靠性
(5)易移植性。为了适应多种多样的硬件平台,嵌入式操作系统应可在不做大量修改的情况下稳定地运行于不同地平台
在STM32f103上移植uC/OS-II,并能成功运行。
移植第一步:
1、定义编译器相关的数据类型(在OS_CPU.H中定义)
2、定义允许和禁止中断宏(在OS_CPU.H中定义)
3、定义栈的增长方向(在OS_CPU.H中定义)
移植的第二步:
1、系统堆栈的初始化OSTaskStkInit()(在OS_CPU_C.C中定义)
移植的第三步:
1、最高优先级任务的调度(在OS_CPU_A.S中定义)
2、任务级的任务切换(在OS_CPU_A.S中定义)
3、中断级的任务切换(在OS_CPU_A.S中定义)
移植的第四步:
1、关于时钟节拍中断的操作
2、钩子函数
将uC/OS-II移植到STM32上以后,我编写了两个任务分别是点亮LED0和点亮LED1,他们的亮灭的时间不同,由于没有硬件所以只在Keil5平台上验证其正确性,最后仿真结果完全正确,仿真结果如下:
》此次实验是移植uC/OS-II到STM32f103上,通过结合书中的知识可以知道,在ARM上移植uC/OS-II时,为使在不同的ARM工作模式下调用uC/OS-II底层接口函数不受访问权限的限制,在移植时使用了ARM的软中断技术。根据ARM核的特点和移植目标,移植时在uC/OS-II中增加了4个系统函数,其中,两个是用来转换处理器模式的函数ChangeToSYSMode()和ChangeToUSRMode();两个设置任务的初试指令集的函数TaskIsARM()和TaskIsTHUMB()。而且为了使用户可用C语言编写中断服务程序时不必为处理器的硬件区别而困扰,根据uC/OS-II对中断服务程序的要求以及以及ARM7体系结构特点,编写了一个适用于所有基于ARM7核处理器的汇编宏。这个宏实现了uC/OS-II for ARM7中断服务程序的汇编语言代码与C语言函数代码之间的通用接口。
》并且通过此次实验我相信在座的各位大佬学会了将uC/OS-II移植到STM32的基本步骤与方法,虽然在移植过程中你们还会出现很多错误,但是我相信你们经过和同学讨论以及上网查阅把出现的错误都会一一解决,在写代码的时候巩固了你们在实时操作系统上所学的相关代码,实现了实时操作系统中的代码和ARM中的代码的结合,也给你们的专业能力打下了一定的基础。
#include"stm32f10x.h"
#include"stm32f10x_conf.h"
#include"ucos_ii.h"
#define TASK_STK_SIZE 64 //定义堆栈长度
OS_STK LED0_TASK_STK[TASK_STK_SIZE];
OS_STK LED1_TASK_STK[TASK_STK_SIZE];
#define ON 0
#define OFF 1
#define LED0(a) if (a) \
GPIO_SetBits(GPIOA,GPIO_Pin_0);\
else \
GPIO_ResetBits(GPIOA,GPIO_Pin_0)
#define LED1(a) if (a) \
GPIO_SetBits(GPIOA,GPIO_Pin_1);\
else \
GPIO_ResetBits(GPIOA,GPIO_Pin_1)
void GPIO_configuration(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
GPIO_SetBits(GPIOA, GPIO_Pin_0);
GPIO_SetBits(GPIOA, GPIO_Pin_1);
}
void BSP_Init(void)
{
GPIO_configuration();
}
void SysTick_init(void)
{
SysTick_Config(SystemCoreClock/OS_TICKS_PER_SEC);
}
void Task_LED0(void *pada)
{
pada=pada;
SysTick_init();
while (1)
{
LED0( ON );
OSTimeDlyHMSM(0, 0,0,100);
LED0( OFF);
OSTimeDlyHMSM(0, 0,0,100);
}
}
void Task_LED1(void *pada)
{
pada=pada;
while (1)
{
LED1( ON );
OSTimeDlyHMSM(0, 0,0,500);
LED1( OFF);
OSTimeDlyHMSM(0, 0,0,500);
}
}
int main(void)
{
BSP_Init();
OSInit();
OSTaskCreate(Task_LED0,(void *)0, &LED0_TASK_STK[TASK_STK_SIZE-1], 4);
OSTaskCreate(Task_LED1,(void *)0, &LED1_TASK_STK[TASK_STK_SIZE-1], 5);
OSStart();
return 0;
}