1、实现功能
通过两个按键任务分别控制不同的点灯案例
创建 4 个任务:在点灯任务的基础上在创建两个按键任务:
Task_led:间隔 500ms 闪烁 LE1;
Task_led2:间隔 1000ms 闪烁 LED2;
Task_key:如果 Task_led存在,则按下 KEY 后删除 Task_led,否则创建 Task_led;
Task_key2:如果 Task_led2正常运行,则按下 KEY2 后挂起 Task_led2,否则恢复 Task_led2
2、CubeMX配置
2.1 GPIO初始化
2.2 任务建立
3、KEIL 5代码编写
3.1 main.h初始化
#define LED_ON HAL_GPIO_WritePin(LED_GPIO_Port,LED_Pin,GPIO_PIN_SET)
#define LED_OFF HAL_GPIO_WritePin(LED_GPIO_Port,LED_Pin,GPIO_PIN_RESET)
#define LED2_ON HAL_GPIO_WritePin(LED2_GPIO_Port,LED2_Pin,GPIO_PIN_RESET)
#define LED2_OFF HAL_GPIO_WritePin(LED2_GPIO_Port,LED2_Pin,GPIO_PIN_SET)
#define key HAL_GPIO_ReadPin(key_GPIO_Port,key_Pin)
#define key2 HAL_GPIO_ReadPin(key2_GPIO_Port,key2_Pin)
3.2 功能实现
任务1:
void StartTask_led(void const * argument)
{
/* USER CODE BEGIN StartTask_led */
/* Infinite loop */
for(;;)
{
LED_ON;
osDelay(500);
LED_OFF;
osDelay(500);
}
/* USER CODE END StartTask_led */
}
任务2:
void StartTask_led2(void const * argument)
{
/* USER CODE BEGIN StartTask_led2 */
/* Infinite loop */
for(;;)
{
LED2_ON;
osDelay(1000);
LED2_OFF;
osDelay(1000);
}
/* USER CODE END StartTask_led2 */
}
任务3:
void StartTask_key(void const * argument)
{
/* USER CODE BEGIN StartTask_key */
/* Infinite loop */
for(;;)
{
if(key==0){
osDelay(20);
if(key==0){
printf("key按下\r\n");
if(Task_ledHandle == NULL){
osThreadDef(Task_led, StartTask_led, osPriorityNormal, 0, 128);
Task_ledHandle = osThreadCreate(osThread(Task_led), NULL);
if(Task_ledHandle != NULL){
printf("task_1成功\r\n");
}
}
else{
printf("detele\r\n");
osThreadTerminate(Task_ledHandle);
Task_ledHandle = NULL;
}
}
while(key==0);
}
osDelay(10);
}
/* USER CODE END StartTask_key */
}
任务4:
void StartTask_key2(void const * argument)
{
/* USER CODE BEGIN StartTask_key2 */
static int flag = 0;
/* Infinite loop */
for(;;)
{
if(key2==0){
osDelay(20);
if(key2==0){
printf("key2按下\r\n");
if(flag == 0){
osThreadSuspend(Task_led2Handle);
printf("任务2以暂停\r\n");
flag=1;
}
else{
osThreadResume(Task_led2Handle);
printf("任务2恢复\r\n");
flag=0;
}
}
while(key2==0);
}
osDelay(10);
}
/* USER CODE END StartTask_key2 */
}
整个freertos.c代码如下:
/* USER CODE BEGIN Header */
/**
******************************************************************************
* File Name : freertos.c
* Description : Code for freertos applications
******************************************************************************
* @attention
*
* Copyright (c) 2023 STMicroelectronics.
* All rights reserved.
*
* This software is licensed under terms that can be found in the LICENSE file
* in the root directory of this software component.
* If no LICENSE file comes with this software, it is provided AS-IS.
*
******************************************************************************
*/
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "FreeRTOS.h"
#include "task.h"
#include "main.h"
#include "cmsis_os.h"
/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */
/* USER CODE END Includes */
/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */
/* USER CODE END PTD */
/* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD */
/* USER CODE END PD */
/* Private macro -------------------------------------------------------------*/
/* USER CODE BEGIN PM */
/* USER CODE END PM */
/* Private variables ---------------------------------------------------------*/
/* USER CODE BEGIN Variables */
/* USER CODE END Variables */
osThreadId Task_ledHandle;
osThreadId Task_led2Handle;
osThreadId Task_keyHandle;
osThreadId Task_key2Handle;
/* Private function prototypes -----------------------------------------------*/
/* USER CODE BEGIN FunctionPrototypes */
/* USER CODE END FunctionPrototypes */
void StartTask_led(void const * argument);
void StartTask_led2(void const * argument);
void StartTask_key(void const * argument);
void StartTask_key2(void const * argument);
void MX_FREERTOS_Init(void); /* (MISRA C 2004 rule 8.1) */
/* GetIdleTaskMemory prototype (linked to static allocation support) */
void vApplicationGetIdleTaskMemory( StaticTask_t **ppxIdleTaskTCBBuffer, StackType_t **ppxIdleTaskStackBuffer, uint32_t *pulIdleTaskStackSize );
/* USER CODE BEGIN GET_IDLE_TASK_MEMORY */
static StaticTask_t xIdleTaskTCBBuffer;
static StackType_t xIdleStack[configMINIMAL_STACK_SIZE];
void vApplicationGetIdleTaskMemory( StaticTask_t **ppxIdleTaskTCBBuffer, StackType_t **ppxIdleTaskStackBuffer, uint32_t *pulIdleTaskStackSize )
{
*ppxIdleTaskTCBBuffer = &xIdleTaskTCBBuffer;
*ppxIdleTaskStackBuffer = &xIdleStack[0];
*pulIdleTaskStackSize = configMINIMAL_STACK_SIZE;
/* place for user code */
}
/* USER CODE END GET_IDLE_TASK_MEMORY */
/**
* @brief FreeRTOS initialization
* @param None
* @retval None
*/
void MX_FREERTOS_Init(void) {
/* USER CODE BEGIN Init */
/* USER CODE END Init */
/* USER CODE BEGIN RTOS_MUTEX */
/* add mutexes, ... */
/* USER CODE END RTOS_MUTEX */
/* USER CODE BEGIN RTOS_SEMAPHORES */
/* add semaphores, ... */
/* USER CODE END RTOS_SEMAPHORES */
/* USER CODE BEGIN RTOS_TIMERS */
/* start timers, add new ones, ... */
/* USER CODE END RTOS_TIMERS */
/* USER CODE BEGIN RTOS_QUEUES */
/* add queues, ... */
/* USER CODE END RTOS_QUEUES */
/* Create the thread(s) */
/* definition and creation of Task_led */
osThreadDef(Task_led, StartTask_led, osPriorityNormal, 0, 128);
Task_ledHandle = osThreadCreate(osThread(Task_led), NULL);
/* definition and creation of Task_led2 */
osThreadDef(Task_led2, StartTask_led2, osPriorityBelowNormal, 0, 128);
Task_led2Handle = osThreadCreate(osThread(Task_led2), NULL);
/* definition and creation of Task_key */
osThreadDef(Task_key, StartTask_key, osPriorityNormal, 0, 128);
Task_keyHandle = osThreadCreate(osThread(Task_key), NULL);
/* definition and creation of Task_key2 */
osThreadDef(Task_key2, StartTask_key2, osPriorityNormal, 0, 128);
Task_key2Handle = osThreadCreate(osThread(Task_key2), NULL);
/* USER CODE BEGIN RTOS_THREADS */
/* add threads, ... */
/* USER CODE END RTOS_THREADS */
}
/* USER CODE BEGIN Header_StartTask_led */
/**
* @brief Function implementing the Task_led thread.
* @param argument: Not used
* @retval None
*/
/* USER CODE END Header_StartTask_led */
void StartTask_led(void const * argument)
{
/* USER CODE BEGIN StartTask_led */
/* Infinite loop */
for(;;)
{
LED_ON;
osDelay(500);
LED_OFF;
osDelay(500);
}
/* USER CODE END StartTask_led */
}
/* USER CODE BEGIN Header_StartTask_led2 */
/**
* @brief Function implementing the Task_led2 thread.
* @param argument: Not used
* @retval None
*/
/* USER CODE END Header_StartTask_led2 */
void StartTask_led2(void const * argument)
{
/* USER CODE BEGIN StartTask_led2 */
/* Infinite loop */
for(;;)
{
LED2_ON;
osDelay(1000);
LED2_OFF;
osDelay(1000);
}
/* USER CODE END StartTask_led2 */
}
/* USER CODE BEGIN Header_StartTask_key */
/**
* @brief Function implementing the Task_key thread.
* @param argument: Not used
* @retval None
*/
/* USER CODE END Header_StartTask_key */
void StartTask_key(void const * argument)
{
/* USER CODE BEGIN StartTask_key */
/* Infinite loop */
for(;;)
{
if(key==0){
osDelay(20);
if(key==0){
printf("key按下\r\n");
if(Task_ledHandle == NULL){
osThreadDef(Task_led, StartTask_led, osPriorityNormal, 0, 128);
Task_ledHandle = osThreadCreate(osThread(Task_led), NULL);
if(Task_ledHandle != NULL){
printf("task_1成功\r\n");
}
}
else{
printf("detele\r\n");
osThreadTerminate(Task_ledHandle);
Task_ledHandle = NULL;
}
}
while(key==0);
}
osDelay(10);
}
/* USER CODE END StartTask_key */
}
/* USER CODE BEGIN Header_StartTask_key2 */
/**
* @brief Function implementing the Task_key2 thread.
* @param argument: Not used
* @retval None
*/
/* USER CODE END Header_StartTask_key2 */
void StartTask_key2(void const * argument)
{
/* USER CODE BEGIN StartTask_key2 */
static int flag = 0;
/* Infinite loop */
for(;;)
{
if(key2==0){
osDelay(20);
if(key2==0){
printf("key2按下\r\n");
if(flag == 0){
osThreadSuspend(Task_led2Handle);
printf("任务2以暂停\r\n");
flag=1;
}
else{
osThreadResume(Task_led2Handle);
printf("任务2恢复\r\n");
flag=0;
}
}
while(key2==0);
}
osDelay(10);
}
/* USER CODE END StartTask_key2 */
}
/* Private application code --------------------------------------------------*/
/* USER CODE BEGIN Application */
/* USER CODE END Application */