FreeRTOS. Сигналы.
Сигнал (в теории информации и связи) — носитель информации, используемый для передачи сообщений в системе связи. Сигнал может генерироваться, но его приём не обязателен, в отличие от сообщения, которое рассчитано на принятие принимающей стороной, иначе оно не является сообщением. Сигналом может быть любой физический процесс, параметры которого изменяются (или находятся) в соответствии с передаваемым сообщением. Фактически, сигнал — это асинхронное уведомление процесса о каком-либо событии. Когда сигнал послан процессу, операционная система прерывает выполнение процесса. Если процесс установил собственный обработчик сигнала, операционная система запускает этот обработчик, передав ему информацию о сигнале. Если процесс не установил обработчик, то выполняется обработчик по умолчанию.
Описание
Сигналы используются для запуска состояния выполнения между потоками. Функции управления сигнал в CMSIS-RTOS позволяет вам управлять или ждать сигнальных флагов. Каждый поток имеет до 31 назначенных сигнальных флагов. Максимальное количество сигнальных флагов определено в файле cmsis_os.h (#define) osFeature_Signals.
Поток
может ждать сигнала, который должен быть установлен (с использованием osSignalWait). Используя эту функцию, он переходит в состояние ожидания. Сигналы параметров osSignalWait определяет сигналы, которые необходимы, чтобы перевести поток обратно в состояние готовности. Так же, можно установить один или более флагов в любом другом потоке данных (с использованием osSignalSet). Можно очистить свои собственные сигналы или сигналы других потоков (с использованием osSignalClear).
Когда поток просыпается и возобновляет выполнение, его сигнал-флаги автоматически удаляются.
Работа с сигналами
Вот простой пример, который показывает, как два потока могут общаться друг с другом, используя сигналы:

Пример cвязи события простого сигнала
Следующие шаги необходимы чтобы использовать сигналы. Мы имеем поток 1 (tid_thread1), в нем ожидаем прихода сигнала, вызываем функцию ожидания osSignalWait (0x0001, osWaitForever), ждем сигнала 0x0001. В другом потоке, поток 2, который предполагается, разбудить ожидание потока с помощью функции osSignalSet (tid_thread1, 0x0001), тем самым установив сигнал 0x0001 для потока 1 (tid_thread1). После действия повторяются циклически, для тайм аута используем функцию osDelay (1000), ждать в течение 1 секунды.
Наш пример создает потоки, которые вызывают osSignalWait (поток ожидания сигнала), ждем установления бит 1, а затем бит 2 и переключаем состояние светодиода. SysTick_Handler вызывает функцию “Toggle_Leds”. Toggle_Leds вызывает osSignalSet, чтобы послать сигнал = Бит 1 для потока 2, если статический счетчик > 400 и < 1000 и вызывает osSignalSet отправляя сигнал = Бит 1 | бит 2 для потока 2, если статический счетчик> = 1000. В результате поведение характеризуется следующим образом, светодиод LED переключаем каждые 800 мс.
Необходимо соблюдать осторожность при использовании HAL_Delay (), эта функция обеспечивает точное задержку (в миллисекундах) основная переменная увеличивается на единицу в SysTick ISR. Это означает, что если HAL_Delay () вызывается из периферийного процесса ISR, то прерывание SysTick должны иметь более высокий приоритет (численно меньше) чем окружные прерывания. В противном случае процесс ISR вызывающий будет заблокирован.
Чтобы изменить приоритет прерывания SysTick вы должны использовать функцию HAL_NVIC_SetPriority ().
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "cmsis_os.h"
/* Private typedef -----------------------------------------------------------*/
/* Private define ------------------------------------------------------------*/
#define BIT_0 ( 1 << 0 )
#define BIT_1 ( 1 << 1 )
#define BIT_2 ( 1 << 2 )
/* Private macro -------------------------------------------------------------*/
/* Private variables ---------------------------------------------------------*/
osThreadId LED_ThreadId;
/* Private function prototypes -----------------------------------------------*/
static void LED_Thread(void const *argument);
static void SystemClock_Config(void);
static void CPU_CACHE_Enable(void);
void Toggle_Leds(void);
/* Private functions ---------------------------------------------------------*/
/**
* @brief Main program
* @param None
* @retval None
*/
int main(void)
{
/* Enable the CPU Cache */
CPU_CACHE_Enable();
/* STM32F7xx HAL library initialization:
- Configure the Flash ART accelerator on ITCM interface
- Configure the Systick to generate an interrupt each 1 msec
- Set NVIC Group Priority to 4
- Low Level Initialization
*/
HAL_Init();
/* Configure the system clock to 216 Mhz */
SystemClock_Config();
/* Initialize LEDs */
BSP_LED_Init(LED1);
/* Thread 1 definition */
osThreadDef(LED, LED_Thread, osPriorityNormal, 0, configMINIMAL_STACK_SIZE);
/* Start thread */
LED_ThreadId = osThreadCreate(osThread(LED), NULL);
/* Start scheduler */
osKernelStart();
/* We should never get here as control is now taken by the scheduler */
for(;;);
}
/**
* @brief Toggle LED1 thread
* @param argument not used
* @retval None
*/
static void LED_Thread(void const *argument)
{
(void) argument;
osEvent event;
for(;;)
{
event = osSignalWait( BIT_1 | BIT_2, osWaitForever);
if(event.value.signals == (BIT_1 | BIT_2))
{
BSP_LED_Toggle(LED1);
}
}
}
/**
* @brief System Clock Configuration
* The system Clock is configured as follow :
* System Clock source = PLL (HSE)
* SYSCLK(Hz) = 216000000
* HCLK(Hz) = 216000000
* AHB Prescaler = 1
* APB1 Prescaler = 4
* APB2 Prescaler = 2
* HSE Frequency(Hz) = 25000000
* PLL_M = 25
* PLL_N = 432
* PLL_P = 2
* PLL_Q = 9
* VDD(V) = 3.3
* Main regulator output voltage = Scale1 mode
* Flash Latency(WS) = 7
* @param None
* @retval None
*/
static void SystemClock_Config(void)
{
RCC_ClkInitTypeDef RCC_ClkInitStruct;
RCC_OscInitTypeDef RCC_OscInitStruct;
/* Enable HSE Oscillator and activate PLL with HSE as source */
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
RCC_OscInitStruct.HSEState = RCC_HSE_ON;
RCC_OscInitStruct.HSIState = RCC_HSI_OFF;
RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
RCC_OscInitStruct.PLL.PLLM = 25;
RCC_OscInitStruct.PLL.PLLN = 432;
RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2;
RCC_OscInitStruct.PLL.PLLQ = 9;
HAL_RCC_OscConfig(&RCC_OscInitStruct);
/* Select PLL as system clock source and configure the HCLK, PCLK1 and PCLK2
clocks dividers */
RCC_ClkInitStruct.ClockType = (RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2);
RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV4;
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2;
HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_7);
}
/**
* @brief Toggles LEDs to show user input state.
* @param None
* @retval None
*/
void Toggle_Leds(void)
{
static uint32_t ticks = 0;
if((ticks++ > 400) && (ticks < 1000))
{
osSignalSet( LED_ThreadId, BIT_1);
}
else if(ticks++ >= 1000)
{
osSignalSet( LED_ThreadId, BIT_1 | BIT_2 );
ticks = 0;
}
}
#ifdef USE_FULL_ASSERT
/**
* @brief Reports the name of the source file and the source line number
* where the assert_param error has occurred.
* @param file: pointer to the source file name
* @param line: assert_param error line source number
* @retval None
*/
void assert_failed(uint8_t* file, uint32_t line)
{
/* User can add his own implementation to report the file name and line number,
ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
/* Infinite loop */
while (1)
{}
}
#endif
/**
* @brief CPU L1-Cache enable.
* @param None
* @retval None
*/
static void CPU_CACHE_Enable(void)
{
/* Enable I-Cache */
SCB_EnableICache();
/* Enable D-Cache */
SCB_EnableDCache();
}
/**
* @brief This function handles SysTick Handler.
* @param None
* @retval None
*/
void SysTick_Handler(void)
{
HAL_IncTick();
if (xTaskGetSchedulerState() != taskSCHEDULER_NOT_STARTED)
{
osSystickHandler();
Toggle_Leds();
}
}
Старт ARM. RTOS часть 1-ая. STM32F4 и SAM3N.
Старт ARM. RTOS часть 2-ая.
Старт ARM. RTOS часть 3-ая. Очереди.
Старт ARM. RTOS часть 4-ая. Семафоры.
Старт ARM. RTOS часть 5-ая. Мьютексы.
Старт ARM. RTOS часть 6-ая. Сопрограмма.
Старт ARM. RTOS часть 7-ая. Программный таймер.
Старт ARM. RTOS часть 8-ая. Программная очередь почты (mail queue).
One thought on “FreeRTOS. Сигналы.”
Господа! Подскажите пожалуйста) плата STM32 F746G disco , по мануалу для для F429 сделал в Gui builder окошко с кнопкой, как сделать так чтоб по нажатию на кнопку открывалось новое окно?? И вообще есть какая – нибудь литература или ресурсы по работе с Gui builder?
Comments are closed.