STM32CubeMX FreeRTOS 互斥锁

一、CubeMX配置

 时钟配置

LED板载小灯配置

串口一配置

freertos配置

 生成工程

二、互斥锁(Mutex)

  • 特点:
    • 互斥锁是一种基本的同步原语,用于保护临界区,确保在同一时刻只有一个线程可以进入临界区。
    • 当一个线程持有互斥锁时,其他线程在尝试获得同一互斥锁时会被阻塞,直到该锁被释放。
    • 同一线程多次尝试获得相同的互斥锁会导致死锁。 

FreeRTOSConfig.h文件通常位于Middlewares/Third_Party/FreeRTOS/Source/include目录下。

在FreeRTOSConfig.h文件中,确保configUSE_MUTEXES宏被设置为1,以启用互斥锁。

#define configUSE_MUTEXES      1

添加头文件

/* USER CODE BEGIN Includes */
#include "semphr.h"
#include "stdio.h"
#include "usart.h"
/* USER CODE END Includes */

配置串口重定向,选好记得保存

 

/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */

#include <stdio.h>
int fputc(int ch,FILE *f)
{
    HAL_UART_Transmit(&huart1,(uint8_t*)&ch,1,0xFFFF);
    return ch;
}
/* USER CODE END PTD */

创建句柄

/* USER CODE BEGIN PD */

// 定义一个互斥锁句柄
SemaphoreHandle_t xMutex;

/* USER CODE END PD */

创建互斥锁

  /* USER CODE BEGIN RTOS_MUTEX */
  /* add mutexes, ... */
    // 创建互斥锁
    xMutex = xSemaphoreCreateMutex();
  /* USER CODE END RTOS_MUTEX */

任务一抢占互斥锁

/* USER CODE END Header_StartDefaultTask */
void StartDefaultTask(void *argument)
{
  /* USER CODE BEGIN StartDefaultTask */
  /* Infinite loop */
  for(;;)
  {
       // 尝试获取互斥锁
        if (xSemaphoreTake(xMutex, portMAX_DELAY)) {
            printf("Task1 get ok\r\n");
            xSemaphoreGive(xMutex); // 释放互斥锁
        }
        vTaskDelay(pdMS_TO_TICKS(1000)); // 任务延时
    osDelay(1);
  }
  /* USER CODE END StartDefaultTask */
}

任务二抢占互斥锁

/* USER CODE END Header_StartTask02 */
void StartTask02(void *argument)
{
  /* USER CODE BEGIN StartTask02 */
  /* Infinite loop */
  for(;;)
  {
      // 尝试获取互斥锁
        if (xSemaphoreTake(xMutex, portMAX_DELAY)) {
           printf("Task2 get ok\r\n");
            xSemaphoreGive(xMutex); // 释放互斥锁
        }
        vTaskDelay(pdMS_TO_TICKS(1000)); // 任务延时
    osDelay(1);
  }
  /* USER CODE END StartTask02 */
}

运行效果  同一时刻只有一个任务

三、递归互斥锁(Recursive Mutex)

特点:

  • 递归互斥锁允许同一线程在持有锁的情况下多次获取该锁,而不会造成死锁。
  • 每次获取锁都需要对应的释放,只有当所有获取和释放的操作平衡时,其他线程才能获得该锁。
  • 允许嵌套,即同一线程在多层函数调用中可以多次获取相同的递归互斥锁。

 FreeRTOSConfig.h文件通常位于Middlewares/Third_Party/FreeRTOS/Source/include目录下。

在FreeRTOSConfig.h文件中,确保configUSE_MUTEXES宏被设置为1,以启用互斥锁。

#define configUSE_MUTEXES      1

添加头文件

/* USER CODE BEGIN Includes */
#include "semphr.h"
#include "stdio.h"
#include "usart.h"
/* USER CODE END Includes */

配置串口重定向,选好记得保存

 

/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */

#include <stdio.h>
int fputc(int ch,FILE *f)
{
    HAL_UART_Transmit(&huart1,(uint8_t*)&ch,1,0xFFFF);
    return ch;
}
/* USER CODE END PTD */

创建句柄

/* USER CODE BEGIN PD */

// 定义一个递归互斥锁句柄
SemaphoreHandle_t xRecursiveMutex;

/* USER CODE END PD */

创建递归互斥锁

  /* USER CODE BEGIN RTOS_MUTEX */
  /* add mutexes, ... */
    // 创建递归互斥锁
    xRecursiveMutex = xSemaphoreCreateRecursiveMutex();
  /* USER CODE END RTOS_MUTEX */

任务一抢占

/* USER CODE END Header_StartDefaultTask */
void StartDefaultTask(void *argument)
{
  /* USER CODE BEGIN StartDefaultTask */
  /* Infinite loop */
  for(;;)
  {
        // 尝试获取递归互斥锁
        if (xSemaphoreTakeRecursive(xRecursiveMutex, portMAX_DELAY)) {
            printf("Task1 get ok\r\n");
            xSemaphoreGiveRecursive(xRecursiveMutex); // 释放递归互斥锁
        }
        vTaskDelay(pdMS_TO_TICKS(1000)); // 任务延时
    osDelay(1);
  }
  /* USER CODE END StartDefaultTask */
}

任务二抢占

/* USER CODE END Header_StartTask02 */
void StartTask02(void *argument)
{
  /* USER CODE BEGIN StartTask02 */
  /* Infinite loop */
  for(;;)
  {
        // 尝试获取递归互斥锁
        if (xSemaphoreTakeRecursive(xRecursiveMutex, portMAX_DELAY)) {
            printf("Task2 get ok\r\n");
            xSemaphoreGiveRecursive(xRecursiveMutex); // 释放递归互斥锁
        }
        vTaskDelay(pdMS_TO_TICKS(1000)); // 任务延时
    osDelay(1);
  }
  /* USER CODE END StartTask02 */
}
  1. 递归调用: RecursiveTask(pvParameters) 语句表示调用当前函数,即递归调用。这是为了模拟同一线程在递归调用中多次获取和释放递归互斥锁。

  2. xSemaphoreGiveRecursive(xRecursiveMutex) 在递归调用返回后,表示退出临界区并释放递归互斥锁。

  3. vTaskDelay(pdMS_TO_TICKS(1000)) 为了简化演示,这里使用了延时,模拟任务执行一段时间后再次尝试获取锁。

  4. vTaskDelete(NULL) 在任务执行结束时,调用 vTaskDelete(NULL) 表示删除当前任务。任务执行结束后,相应的资源将被释放。 

运行效果 

在使用STM32CubeMX配置FreeRTOS时,你可以使用互斥锁来实现多任务之间的互斥访问。互斥锁是一种同步机制,它可以确保在任何给定时间只有一个任务可以访问共享资源。 要在FreeRTOS中使用互斥锁,你需要进行以下步骤: 1. 在STM32CubeMX中打开项目配置。 2. 打开RTE(Run-Time Environment)配置窗口。 3. 在RTE组件中搜索"mutex"或"互斥锁",并选择适合你的微控制器的互斥锁组件。 4. 为你的任务创建互斥锁。 创建互斥锁的代码示例: ```c #include "cmsis_os.h" // 声明互斥锁句柄 osMutexId mutexHandle; // 在任务初始化函数中创建互斥锁 void task_init(void const * argument) { // 创建互斥锁 osMutexDef(mutex); mutexHandle = osMutexCreate(osMutex(mutex)); // ... } // 在任务中使用互斥锁 void task_example(void const * argument) { // 等待获取互斥锁 if(osMutexWait(mutexHandle, osWaitForever) == osOK) { // 临界区代码 // ... // 释放互斥锁 osMutexRelease(mutexHandle); } // ... } ``` 在上面的示例中,首先在任务初始化函数中创建了一个互斥锁,然后在任务中使用了该互斥锁来保护临界区代码。使用`osMutexWait`函数等待获取互斥锁,如果成功获取到互斥锁,则进入临界区代码执行,并在临界区代码执行完毕后使用`osMutexRelease`函数释放互斥锁。 这样就可以确保只有一个任务可以同时访问被保护的临界区代码,从而实现了任务间的互斥访问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

chem4111

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值