基于STM32的虚拟多线程

2018-06-10 12:30:23编辑:什么鱼 关键字:STM32  虚拟多线程

基于STM32虚拟多线程,可以很好的用于裸机程序中,用于模拟小型操作系统的多线程概念。本实例参考了参考TI_BLE协议栈_ZStack协议栈。

#include "Hal_Led/Hal_Led.h"

#include "Hal_delay/delay.h"

#include "Hal_Key/Hal_Key.h"

#include "ringbuffer.h"

#define APP_LED2_BLINK_EVENT   0x0001

#define HAL_LED1_BLINK_EVENT   0x0001



#define TASK_NO_TASK_RUNNING      0xFF


unsigned short Hal_ProcessEvent( unsigned char task_id, unsigned short events );

unsigned short App_ProcessEvent( unsigned char task_id, unsigned short events );



typedef unsigned short  uint16;

typedef unsigned char   uint8;


#define TASK_CNT  2    //定义线程的个数

//定义函数指针

typedef unsigned short (*pTaskEventHandlerFn)( unsigned char task_id, unsigned short events );

//线程函数表

const pTaskEventHandlerFn tasksArr[] =

{

    Hal_ProcessEvent,

    App_ProcessEvent

};

const unsigned char tasksCnt = sizeof( tasksArr ) / sizeof( tasksArr[0] );


//uint16 *tasksEvents;

uint16 tasksEvents[TASK_CNT] = {0};  //每个线程有16位位域空间用于设置事件


static uint8 activeTaskID = 0xFF;  //当前任务ID,指示作用



#define SUCCESS                   0x00

#define FAILURE                   0x01

#define INVALID_TASK              0x02

uint8 osal_set_event( uint8 task_id, uint16 event_flag )

{

    if ( task_id < tasksCnt )

    {

        tasksEvents[task_id] |= event_flag;  // Stuff the event bit(s)

        return ( SUCCESS );

    }

    else

    {

        return ( INVALID_TASK );

    }

}


/**

* @brief 程序入口

* @param none

* @return none

*/

int main(void)

{

    unsigned short taskID = 0;

    uint8 idx = 0;

    SystemInit();  //系统时钟初始化

    delayInit(72); //滴答定时器初始化

    Led_Init();    //LED初始化

    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);

    osal_set_event(0, HAL_LED1_BLINK_EVENT);

    osal_set_event(1, APP_LED2_BLINK_EVENT);

    while(1)

    {

        do

        {

            if(tasksEvents[idx])  //轮训获知哪个线程有事件需要进行处理

            {

                break;

            }

        }

        while (++idx < tasksCnt);


        if (idx < tasksCnt)

        {

            uint16 events;


            events = tasksEvents[idx];

            tasksEvents[idx] = 0;                    // 清除事件数组中的事件


            activeTaskID = idx;

            events = (tasksArr[idx])( idx, events ); //调用线程函数

            activeTaskID = TASK_NO_TASK_RUNNING;



            tasksEvents[idx] |= events;  // 添加未处理的事件到本线程的事件组中

        }

        delayMs(1000);



    }

}


/**

* @brief 应用层处理

* @param none

* @r

*/


unsigned short Hal_ProcessEvent( unsigned char task_id, unsigned short events )

{

    if ( events & HAL_LED1_BLINK_EVENT )

    {

        Led_Reverse(1);


        return events ^ HAL_LED1_BLINK_EVENT;  //清除事件

    }

}



/**

* @brief 硬件控制线程

* @param none

* @r

*/

unsigned short App_ProcessEvent( unsigned char task_id, unsigned short events )

{

    if ( events & APP_LED2_BLINK_EVENT )

    {

        Led_Reverse(2);


        return events ^ APP_LED2_BLINK_EVENT;  //清除事件

    }

}



关键字:STM32  虚拟多线程

来源: eefocus 引用地址:http://www.eeworld.com.cn/mcu/article_2018061039732.html
本网站转载的所有的文章、图片、音频视频文件等资料的版权归版权所有人所有,本站采用的非本站原创文章及图片等内容无法一一联系确认版权者。如果本网所选内容的文章作者及编辑认为其作品不宜公开自由传播,或不应无偿使用,请及时通过电子邮件或电话通知我们,以迅速采取适当措施,避免给双方造成不必要的经济损失。

上一篇:STM32 printf函数打印到串口
下一篇:基于stm32串口环形缓冲队列处理机制—入门级(单字节)

关注eeworld公众号 快捷获取更多信息
关注eeworld公众号
快捷获取更多信息
关注eeworld服务号 享受更多官方福利
关注eeworld服务号
享受更多官方福利

推荐阅读

C51、STM32和S3C2440中断体系的比较

/* 名称:C51、STM32和S3C2440中断体系的比较 说明:对于这三个芯片的中断体系来说,就我目前了解的, 我说:差不多。这里说的差不多,是中断的本质上差不多,都是首先中断源提出申请(比如触发了外部中断,定时器中断等),然后经过中断源屏蔽寄存器,再然后中断优先级的比较,最后CPU相应中断。(我去,这好像又回到了计算机组成原理啊)。不同的是细节,C51作为低级一点的芯片,其中断少,中断能达到的功能也少,相对来说较为简单些(无论是设计还是具体使用)。而STM32和S3C2440来说,其提供的功能较多,设计起来和使用起来要复杂的多(使用的时候得配置好多好多寄存器)。目前总结的,差不多就是这些吧。*/
发表于 2018-07-21 14:40:54

STM32_GPIO之按键输入

/* 名称:STM32_GPIO之按键输入 说明:这个实验是GPIO的输入功能。基本的思路和51单片机差不多。也是 操作相应的IO口,不过不同的是对于独立按键输入实验来说,51单片机是 需要一开始把待检测的IO口设置成高电平,然后检测其何时被拉低。而对 于STM32来说,其GPIO端口就可以设置成上拉输入,即不需要再人为的把对 应的对应的IO口设置成高电平,硬件电路会自动把对应端口引脚拉成高电 平。还有一点关于条件编译要说的:这个ifndef… #endif 一般用在头文件中。书上说为了防止重复编译。就我目前了解的来看,在实际应用中,如果在两个.c源文件中都
发表于 2018-07-21 14:37:58

STM32之利用I2C协议读写EEPROM

/* 名称:STM32之利用I2C协议读写EEPROM 说明: 1.利用STM32来读写EEPROM和C51最大的不同就是,前者是直接使用I2C控制器(硬件方式)来产生所需要的I2C时序,而后者是通过软件方式来产生I2C时序。相对来说,前者使用硬件电路简化了编程的复杂性,用户只要将数据发送到相应的数据寄存器,然后I2C控制器自动按照I2C协议把数据通过SCL和SDA总线发送出去。而后者,你必须按照I2C协议手动产生SCL和SDA的高低电平。当然对于EEPROM来说,它是感受不到发送来的数据是通过硬件电路还是软件方式产生的。2.还有,STM32内部集成的I2C控制器不仅可以产生时序,还包括了一些
发表于 2018-07-21 14:36:16

STM32有关GPIO引脚的一些问题

/* 名称:STM32有关GPIO引脚的一些问题 说明:今天在编写一个键盘扫描程序的时候,出现了一些问题。 有的引脚能读出电平状态、有点引脚确一直读不出状态。改了一下午,查了好久,终于有点眉目了。现简述如下: 我在用引脚的时候,使用的是PB0、PB1、PB2、PB3、PB4…,将其引脚配置的方式是上拉输入。PB2、PB3、PB4一直有问题。PB0、PB1是可以读出状态的(但是只能单独使用)。对于PB2来说,其不仅是PB2,它也是BOOT1引脚。当系统复位之后,开始的几个系统时钟使用的是BOOT1,用来判断系统从何处读取置零。之后,PB2就可以作为普通的IO引脚使用了。由于开发板设置
发表于 2018-07-21 12:22:43

STM32定时器之控制LED灯循环显示

/* 名称:STM32定时器之控制LED灯循环显示 说明:对于STM32来说,其定时器要比C51的定时器要多,功能也更复杂。就数目来说来说,STM32的定时器可以分为基本定时器,通用定时器和高级定时器(还包括Systick、看门口定时器和窗口定时器)。就STM32F103,一般来说,它有2个高级定时器TIM1和TIM8、4个通用定时器TIM2/3/4/5、2个基本定时器TIM6和TIM7。而C51一般只有几个定时器。就功能来说, STM32中: 对于基本定时器,其功能就是和C51差不多,只有定时功能。 对于通用定时器,它的功能有定时、输出比较、输入捕获。虽然目前,我也不知道后面两个
发表于 2018-07-21 11:56:24

STM32之LED按键中断

***************************************************************************************************************************************开发板   :奋斗STM32CPU        :STM32F103开发环境:keil uVsion4
发表于 2018-07-21 11:48:43

小广播

何立民专栏

单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号 电信业务审批[2006]字第258号函 京公海网安备110108001534 Copyright © 2005-2018 EEWORLD.com.cn, Inc. All rights reserved