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.