【FreeRtos教程六】STM32 CubeMx——Mutexes And Recursive Mutexes(互斥量与递归互斥量)

目录

1 互斥量

2 示例程序

2.1例程功能

2.2步骤

2.3实验结果

2.4函数讲解

2.5程序源码


1 互斥量

1.都是为什么要有互斥量

在多任务系统中,任务A正在使用某个资源,还没用完的情况下任务B也来使用的话,就可能导致问题。 比如对于串口,任务A正使用它来打印,在打印过程中任务B也来打印,客户看到的结果就是A、B的信 息混杂在一起。

所以我们希望某一任务在在某一时刻单独占有某一硬件资源,这时候便引入了互斥量。例如,我们怎么独享厕所呢?我们可以上厕所然后把门锁了,完事了自己开锁。使用队列、信号量都可以实现互斥访问,以信号量为例:

  • 信号量初始值为1
  • 任务A想上厕所,"take"信号量成功,它进入厕所
  • 任务B也想上厕所,"take"信号量不成功,等待
  • 任务A用完厕所,"give"信号量;轮到任务B使用

2.既然队列、信号量都可以实现互斥访问,为什么还要引入互斥量

假设任务A、B都想使用串口,A优先级比较低:

  • 任务A获得了串口的信号量
  • 任务B也想使用串口,它将会阻塞、等待A释放信号量
  • 高优先级的任务,被低优先级的任务延迟,这被称为"优先级反转"(priority inversion)

互斥量可以通过"优先级继承",可以很大程度解决"优先级反转"的问题,这也是FreeRTOS中互斥量和二 进制信号量的差别。这点在下面的例程中会被介绍。

3.FreeRtos的互斥锁是代码上约定的习惯

使用互斥量的核心在于:谁上锁,就只能由谁开锁。但是FreeRTOS的互斥锁,并没有在代码上实现这点,即使任务A获得了互斥锁,任务B竟然也可以释放互斥锁。所以为了实现互斥锁,就约定程序员在写代码的时候,那个任务上锁,就由那个任务解锁。

4.递归互斥量

互斥量能被多次获取,同时也要被多次释放。其功能就不在详细介绍。

2 示例程序

2.1 例程功能

按优先级由高到底分别创建一个按键任务(priority:High)、打印任务1(priority:Normal)、打印任务2(priority:Low),打印任务1和2先挂起,由按键任务来解挂起,观察打印任务2在添加获取互斥量程序前后,打印任务1和2执行的先后顺序。

2.2 步骤

配置按键IO口

 配置三个动态任务,优先级依次为High、Normal、Low

 配置一个互斥量

按键按下,唤醒打印任务1和打印任务2,并获取释放互斥量

/* USER CODE END Header_StartTask_KEY0 */
void StartTask_KEY0(void const * argument)
{
  /* USER CODE BEGIN StartTask_KEY0 */
  /* Infinite loop */
  for(;;)
  {
    if(HAL_GPIO_ReadPin(KEY0_GPIO_Port,KEY0_Pin)==0)
    {
        osDelay(10);
        if(HAL_GPIO_ReadPin(KEY0_GPIO_Port,KEY0_Pin)==0)
        {
            osThreadSuspendAll();
            printf("\nKEY_0按下\n");
            osThreadResumeAll();
            osThreadResume(myTask01Handle);
            osThreadResume(myTask02Handle);
            osMutexWait(myMutex01Handle,osWaitForever);//获得互斥量
            osMutexRelease(myMutex01Handle);//释放互斥量
            while(HAL_GPIO_ReadPin(KEY0_GPIO_Port,KEY0_Pin)==0)
            {
                osDelay(10);
            }
        }
    }
    osDelay(1);
 }

循环开始时任务挂起,等待按键按下后任务解挂起并打印字符串

void StartTask01(void const * argument)
{
  /* USER CODE BEGIN StartTask01 */
  /* Infinite loop */
  for(;;)
  {
      osThreadSuspend(NULL);
      osThreadSuspendAll();
      printf("嗨,我是打印任务 1\n");
      osThreadResumeAll();
      osDelay(1);
  }
  /* USER CODE END StartTask01 */
}

循环开始时任务挂起,等待按键按下后任务解挂起并打印字符串。程序osMutexWait()和osMutexRelease()屏蔽与不屏蔽,程序执行的结果有所不同。

/* USER CODE END Header_StartTask02 */
void StartTask02(void const * argument)
{
  /* USER CODE BEGIN StartTask02 */
  /* Infinite loop */
  for(;;)
  {
      //osMutexWait(myMutex01Handle,osWaitForever);//获得互斥量
      osThreadSuspend(NULL);
      osThreadSuspendAll();
      printf("嗨,我是打印任务 2\n");
      osThreadResumeAll();
      //osMutexRelease(myMutex01Handle);//释放互斥量
      osDelay(1);
  }
  /* USER CODE END StartTask02 */
}

2.3 实验结果

程序osMutexWait()和osMutexRelease()屏蔽时,当按键按下后,打印任务1和打印任务2按照优先级顺序先后执行

程序osMutexWait()和osMutexRelease()不屏蔽时,当按键按下后,打印任务2发生优先级继承先打印

实验评析

FreeRtos实现互斥的核心概念是:谁上锁,就由谁来解锁,那为什么不用信号量来实现一定要用互斥量来实现呢?关键在于使用信号量实现互斥会发生优先级的反转,而使用互斥量能通过优先级继承的方式来解决优先级反转的问题。(详情见韦东山的讲解)

板子上电,程序初步运行时,打印任务2获取互斥量后然后挂起,当按键按下,按键任务在解挂打印任务1、2后也想获取互斥量,但此时互斥量被打印任务2占有,这时候FreeRtos就会将打印任务2的优先级提升到和按键任务一样高(Priority:High),使互斥量被尽快释放。

2.4 函数讲解

1 osStatus osMutexWait (osMutexId mutex_id, uint32_t millisec)

/**
@brief Wait until a Mutex becomes available
@param mutex_id      mutex ID obtained by \ref osMutexCreate.
@param millisec      timeout value or 0 in case of no time-out.
@retval  status code that indicates the execution status of the function.
@note   MUST REMAIN UNCHANGED: \b osMutexWait shall be consistent in every CMSIS-RTOS.
*/
osStatus osMutexWait (osMutexId mutex_id, uint32_t millisec)
  • 函数功能:获取某一互斥量
  • 传入参数:1.mutex_id (要获取互斥量量的句柄) 2.millisec(超时等待时间)

2 osStatus osMutexRelease (osMutexId mutex_id)

/**
@brief Release a Mutex that was obtained by \ref osMutexWait
@param mutex_id      mutex ID obtained by \ref osMutexCreate.
@retval  status code that indicates the execution status of the function.
@note   MUST REMAIN UNCHANGED: \b osMutexRelease shall be consistent in every CMSIS-RTOS.
*/
osStatus osMutexRelease (osMutexId mutex_id)
  • 函数功能:释放某一互斥量
  • 传入参数:mutex_id (要释放互斥量量的句柄)

2.5 程序源码

github链接:HaoJosephWen/FreeRtos_personal (github.com)

### 回答1: STM32CubeMX是ST公司提供的一款STM32微控制器的配置工具,可以帮助用户快速配置STM32的外设和中断。FreeRTOS是一个免费的嵌入式实时操作系统。 在STM32CubeMX中使用FreeRTOS需要先在STM32CubeMX中配置好外设并生成代码,然后将FreeRTOS的文件加入工程中,最后在生成的代码中添加FreeRTOS相关的代码即可。 教程步骤: 1. 下载并安装STM32CubeMX 2. 下载FreeRTOS源代码 3. 在STM32CubeMX中配置好外设并生成代码 4. 将FreeRTOS的文件加入工程中 5. 在STM32CubeMX生成的代码中添加FreeRTOS相关的代码 6. 编译并下载程序到STM32微控制器中 具体的实现方法可以参考ST官方提供的教程或者网上的相关资料。 ### 回答2: STM32CubeMX是STMicroelectronics公司开发的一个可视化的嵌入式软件配置工具。它支持多种STM32系列微控制器,可以方便地进行Pinout、时钟、电源、GPIO、外设配置,并生成代码框架。这些框架可以方便地集成在任意IDE中,如IAR、KEIL和Atollic TrueSTUDIO等,以便进行开发和调试。 FreeRTOS是一款基于MIT许可证的实时操作系统(RTOS)。它是免费、开源的,具有许多实用的功能,如任务管理、调度、同步、信号、消息队列等。FreeRTOS支持多种微控制器,如ARM Cortex-M系列、MSP430、PIC32等。 在STM32CubeMX中,可以方便地选择FreeRTOS作为操作系统,在初始化时生成FreeRTOS的代码框架。这些框架包含了FreeRTOS的全部功能,并且集成了STM32的底层驱动程序和HAL库。 生成的FreeRTOS代码框架中包含了三个任务(task):任务A、任务B和任务C。每个任务的优先级从高到低分别为1、2和3。在main函数中,首先进行HAL库的初始化,然后创建三个任务,并启动调度程序。 任务A和任务B分别使用vTaskDelay函数实现1s的间隔,任务C则不停地输出信息。在任务A和任务B等待的过程中,FreeRTOS会让出CPU资源,把执行权交给其他任务。 在代码框架中还包含了一个定时器(Timer)和一个队列(Queue)。这些功能可以在任务之间进行通信和同步。例如,在任务A中使用Queue发送消息,在任务B中使用Queue接收消息。在任务C中使用定时器来定时输出信息。 总之,STM32CubeMXFreeRTOS的结合提供了一种方便、快速、高效的嵌入式软件开发方式。它可以帮助开发人员更好地了解STM32微控制器和FreeRTOS操作系统的功能和特性,提高开发效率和代码质。 ### 回答3: STM32CubeMXSTM32系列芯片的一款图形化配置工具,可以快速生成基于HAL库的项目代码。而FreeRTOS是一款实时操作系统,在嵌入式系统应用中得到广泛应用。本文主要围绕STM32CubeMXFreeRTOS的组合应用教程进行阐述。 在使用STM32CubeMX生成工程时,可以在中间的软件栏中勾选FreeRTOS。此时,代码将自动生成FreeRTOS的头文件和源文件以及部分配置文件。需要注意的是,FreeRTOS的版本也可以在中间的软件栏中进行选择。选择完成后,可以对FreeRTOS进行进一步的配置,包括任务数、栈空间大小、时间片轮询优先级等。这些配置都可以在头文件中手动修改,但使用STM32CubeMX可以大大节省时间。 在生成工程后,需要在主函数中添加创建FreeRTOS任务的代码,这部分的代码可以在头文件中找到。上手难度较低,基本相当于“复制粘贴”。通过FreeRTOS,可以很方便地创建多个任务,并对它们进行优先级调度,达到开发设计需求。 需要注意的是,FreeRTOS的使用需要一定的操作系统知识,例如任务调度、信号等概念。建议在学习使用FreeRTOS前,先对操作系统有一定程度的了解。同时,FreeRTOS的使用还需要注意堆栈空间、资源抢占等问题。如果不妥善处理这些问题,会导致系统死锁、运行错误等问题。 总之,使用STM32CubeMXFreeRTOS,可以大大提高嵌入式系统应用的开发效率,是一款非常实用的工具和操作系统。但是,在使用前需要了解一些必要的操作系统知识,并正确处理堆栈空间、资源抢占等问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值