0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Free RTOS的互斥信号量

汽车电子技术 来源:玩转单片机 作者:Julian 2023-02-10 15:36 次阅读

二进制信号量和互斥量非常相似,但确实有一些细微的区别。互斥体包含优先级继承机制,而二进制信号量没有。这使得二进制信号量成为实现同步(任务之间或任务与中断之间)的更好选择,互斥体成为实现简单互斥的更好选择。

使用互斥信号量时,需要在FreeRTOSConfig.h中加入配置代码

//使用互斥信号量
#define configUSE_MUTEXES          1

创建互斥信号量

SemaphoreHandle_t xSemaphoreCreateMutex( void );

返回值:

NULL:创建信号量失败,因为FreeRTOS堆栈不足。

其它值:信号量创建成功。这个返回值存储着信号量句柄。

释放和获取API函数请看二值信号量那篇推文

实验小例程

#include "stm32f10x.h"
#include 
#include "FreeRTOS.h"
#include "task.h"
#include "semphr.h"

//毫秒级的延时
void Delay_Ms(u16 time)
{    
   u16 i=0;  
   while(time--)
   {
      i=12000;  //自己定义
      while(i--) ;    
   }
}

void LED_Init(void)
{
  GPIO_InitTypeDef GPIO_InitStructure;        //定义结构体变量
  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);  //开启时钟
  
  GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;            //选择你要设置的IO口
  GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;      //设置推挽输出模式
  GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;     //设置传输速率
  GPIO_Init(GPIOC,&GPIO_InitStructure);                //初始化GPIO
  
  GPIO_SetBits(GPIOC,GPIO_Pin_0);             //将LED端口拉高,熄灭LED
}

void KEY_Init(void)
{
  GPIO_InitTypeDef GPIO_InitStructure; //定义结构体变量  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOE,ENABLE);
  
  GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;     //选择你要设置的IO口
  GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPD;//下拉输入  
  GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;     //设置传输速率
  GPIO_Init(GPIOA,&GPIO_InitStructure);      /* 初始化GPIO */
  
  GPIO_InitStructure.GPIO_Pin=GPIO_Pin_3|GPIO_Pin_2|GPIO_Pin_4;
  GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;  //上拉输入
  GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  GPIO_Init(GPIOE,&GPIO_InitStructure);
}


void USART_init(uint32_t bound)
{
  GPIO_InitTypeDef GPIO_InitStruct;   //定义GPIO结构体变量
  USART_InitTypeDef USART_InitStruct;   //定义串口结构体变量
  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_USART1,ENABLE);   //使能GPIOC的时钟
  
  GPIO_InitStruct.GPIO_Pin=GPIO_Pin_9;   //配置TX引脚
  GPIO_InitStruct.GPIO_Mode=GPIO_Mode_AF_PP;   //配置PA9为复用推挽输出
  GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;   //配置PA9速率
  GPIO_Init(GPIOA,&GPIO_InitStruct);   //GPIO初始化函数
  
  GPIO_InitStruct.GPIO_Pin=GPIO_Pin_10;   //配置RX引脚
  GPIO_InitStruct.GPIO_Mode=GPIO_Mode_IN_FLOATING;   //配置PA10为浮空输入
  GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;   //配置PA10速率
  GPIO_Init(GPIOA,&GPIO_InitStruct);   //GPIO初始化函数
  
  
  USART_InitStruct.USART_Mode=USART_Mode_Tx|USART_Mode_Rx;   //发送接收模式
  USART_InitStruct.USART_Parity=USART_Parity_No;   //无奇偶校验
  USART_InitStruct.USART_BaudRate=bound;   //波特率
  USART_InitStruct.USART_StopBits=USART_StopBits_1;   //停止位1位
  USART_InitStruct.USART_WordLength=USART_WordLength_8b;   //字长8位
  USART_InitStruct.USART_HardwareFlowControl=USART_HardwareFlowControl_None;   //无硬件数据流控制
  USART_Init(USART1,&USART_InitStruct);   //串口初始化函数
  
  USART_Cmd(USART1,ENABLE);   //使能USART1
}

int fputc(int ch,FILE *f)   //printf重定向函数
{
  USART_SendData(USART1,(uint8_t)ch);   //发送一字节数据
  while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);   //等待发送完成
  return ch;
}


#define START_TASK_PRIO 5      //任务优先级
#define START_STK_SIZE 128      //任务堆栈大小
TaskHandle_t StartTask_Handler;   //任务句柄
void Start_Task(void *pvParameters);//任务函数

#define Low_TASK_PRIO 2       //任务优先级
#define Low_STK_SIZE 50       //任务堆栈大小
TaskHandle_t LowTask_Handler;     //任务句柄
void Low_Task(void *p_arg);     //任务函数

#define Med_TASK_PRIO 3       //任务优先级
#define Med_STK_SIZE 50       //任务堆栈大小
TaskHandle_t MedTask_Handler;     //任务句柄
void Med_Task(void *p_arg);     //任务函数

#define High_TASK_PRIO 4       //任务优先级
#define High_STK_SIZE 50       //任务堆栈大小
TaskHandle_t HighTask_Handler;     //任务句柄
void High_Task(void *p_arg);     //任务函数

SemaphoreHandle_t Mutex_Handle =NULL;  //二值信号量句柄

int main( void ) 
{
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);//设置系统中断优先级分组 4
  
  LED_Init(); //初始化 LED
  KEY_Init();
  USART_init(9600);
  
  //创建开始任务
  xTaskCreate(
    (TaskFunction_t )Start_Task,     //任务函数
    (const char* )"Start_Task",     //任务名称
    (uint16_t )START_STK_SIZE,       //任务堆栈大小
    (void* )NULL,             //传递给任务函数的参数
    (UBaseType_t )START_TASK_PRIO,     //任务优先级
    (TaskHandle_t* )&StartTask_Handler  //任务句柄 
  );
  vTaskStartScheduler();  //开启调度
}

//开始任务函数
void Start_Task(void *pvParameters)
{
  taskENTER_CRITICAL();   //进入临界区
   /* 创建Test_Queue */
  Mutex_Handle = xSemaphoreCreateMutex();
  if(Mutex_Handle != NULL)
  {
    xSemaphoreGive(Mutex_Handle);//释放信号量
  }
  //创建 Low 任务
  xTaskCreate(
    (TaskFunction_t )Low_Task, 
    (const char* )"Low_Task", 
    (uint16_t )Low_STK_SIZE, 
    (void* )NULL,
    (UBaseType_t )Low_TASK_PRIO,
    (TaskHandle_t* )&LowTask_Handler
  );
  //创建 Med 任务
  xTaskCreate(
    (TaskFunction_t )Med_Task, 
    (const char* )"Med_Task", 
    (uint16_t )Med_STK_SIZE, 
    (void* )NULL,
    (UBaseType_t )Med_TASK_PRIO,
    (TaskHandle_t* )&MedTask_Handler
  );
  //创建 High 任务
  xTaskCreate(
    (TaskFunction_t )High_Task, 
    (const char* )"High_Task", 
    (uint16_t )High_STK_SIZE, 
    (void* )NULL,
    (UBaseType_t )High_TASK_PRIO,
    (TaskHandle_t* )&HighTask_Handler
  );
  vTaskDelete(StartTask_Handler); //删除开始任务
  taskEXIT_CRITICAL();   //退出临界区
}


void Low_Task(void *pvParameters)
{
  int count = 0;
  while(1)
  {
    printf("Low正在等待n");
    xSemaphoreTake(Mutex_Handle,portMAX_DELAY);
    printf("Low获取成功%d次n",++count);
    Delay_Ms(5000);
    xSemaphoreGive(Mutex_Handle);//释放信号量
    vTaskDelay(50);
  }
}

void Med_Task(void *pvParameters)
{
  //BaseType_t xReturn = NULL;
  while(1)
  {
    printf("正在运行n");
    vTaskDelay(50);
  }
}

void High_Task(void *pvParameters)
{
  int count = 0;
  while(1)
  {
    printf("High正在等待n");
    xSemaphoreTake(Mutex_Handle,portMAX_DELAY);
    printf("High获取成功%d次n",++count);
    xSemaphoreGive(Mutex_Handle);//释放信号量
    vTaskDelay(50);
  }
}

实验现象


poYBAGPl88aAQierAADP34qVgyk228.png

解读:互斥信号量,其实就是将Low任务的优先级和High任务的优先级变成了一样的优先级(短暂拉高最低优先级任务),从而解决优先级翻转问题

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 二进制
    +关注

    关注

    2

    文章

    707

    浏览量

    41250
  • 继承机制
    +关注

    关注

    0

    文章

    2

    浏览量

    5664
  • 信号量
    +关注

    关注

    0

    文章

    53

    浏览量

    8257
收藏 人收藏

    评论

    相关推荐

    UCOS-II:对于信号量互斥信号量,事件标志组的个人理解-转

    。 ucos中提供了好几个用于同步事件以及共享资源访问的机制,目前我看明白的有信号量互斥信号量,事件标志组。下面谈谈自己对他们的理解:1.互斥信号
    发表于 12-10 21:16

    【安富莱】【RTX操作系统教程】第15章 互斥信号量

    总结 15.1互斥信号量15.1.1互斥信号量的概念及其作用 互斥信号量就是
    发表于 01-30 17:40

    转:第23章 FreeRTOS互斥信号量

    本章节讲解FreeRTOS重要的资源共享机制---互斥信号量(Mutex,即MutualExclusion的缩写)。注意,建议初学者学习完前两个章节的信号量后再学习本章节的互斥
    发表于 09-06 14:58

    第15章 互斥信号量

    15.1 互斥信号量15.1.1互斥信号量的概念及其作用 互斥信号量就是
    发表于 10-06 16:40

    ucos中对信号量互斥信号量、事件标志组的理解

    ucos中对信号量互斥信号量、事件标志组的理解http://bbs.edu118.com/forum.php?mod=viewthread&tid=268&fromuid=204
    发表于 08-23 10:35

    信号量互斥锁、自旋锁

    信号量互斥锁、自旋锁http://bbs.edu118.com/forum.php?mod=viewthread&tid=488&fromuid=231(出处: 信盈达IT技术社
    发表于 08-29 09:48

    信号量互斥信号量该怎么选择?

    既然说信号量可能会导致优先级反转,那全都在工程里使用互斥信号不就行了?还要信号量干啥?大家一起用互斥信号
    发表于 08-26 03:14

    关于UCOSIII的信号量互斥信号量的理解?

    信号量。如果其它任务中有请求信号量,且该任务优先级高于当前任务优先级,进行任务切换;如果其它任务中当前没有请求此信号量,或该任务优先级低于当前任务优先级,不进行任务切换?请求互斥
    发表于 03-13 00:11

    信号量互斥信号量理解

    在UCOSIII中,信号量如果要PEND的话,那这个信号量的cnt必须大于等于1才可以(需要在创建的时候设置第三个参数cnt为1或者,先POST一下才可以)。这个理解对吗?互斥信号量
    发表于 04-21 02:46

    例程使用互斥信号量初始化如何设置?

    OS_MUTEXTEST_MUTEX; //定义一个互斥信号量//创建一个互斥信号量OSMutexCreate((OS_MUTEX*)&TEST_MUTEX, (CPU_CHAR
    发表于 06-02 16:22

    互斥源码分析测试

    文章目录互斥源码分析测试参考资料:RTT官网文档关键字:分析RT-Thread源码、stm32、RTOS互斥
    发表于 08-24 06:01

    信号量是什么?信号量怎么运作

    信号量信号量简介二值信号量计数信号量应用场景二值信号量怎么运作计数信号量怎么运作
    发表于 01-05 08:09

    UCOS扩展例程-UCOSIII互斥信号量

    UCOS扩展例程-UCOSIII互斥信号量
    发表于 12-14 17:24 27次下载

    详解互斥信号量的概念和运行

    1 、互 斥 信 号 量 1.1 互斥信号量的概念及其作用 互斥信号量的主要作用是对资源实现互斥访问,使用二值
    的头像 发表于 10-22 11:57 1w次阅读
    详解<b class='flag-5'>互斥</b><b class='flag-5'>信号量</b>的概念和运行

    Free RTOS的计数型信号量

    上篇讲解了二值信号量,二值信号量只能判断有无,而不能确定事件发生的次数,因此我们为了确定事件的次数引入了计数型信号量
    的头像 发表于 02-10 15:29 710次阅读
    <b class='flag-5'>Free</b> <b class='flag-5'>RTOS</b>的计数型<b class='flag-5'>信号量</b>