当前位置:网站首页>Esp32 message queue using FreeRTOS
Esp32 message queue using FreeRTOS
2022-04-23 01:50:00 【Wireless_ Link】
zero . Statement
We will continue to update this column in serial form , This column plans to update as follows :
Chapter one :ESP-IDF Basic introduction , It mainly involves modules , chip , Introduction to the development board , Environment building , Program compilation and download , Start the process and other basic operations , Make you right ESP-IDF Development has an overall understanding , Lay a foundation for our follow-up study !
Second articles :ESP32-IDF Introduction to peripheral drivers , Mainly based on esp-idf The existing driver, Provide drivers for various peripherals , such as LED,OLED,SPI LCD,TOUCH, infrared ,Codec ic wait , In this article , We're not just doing peripheral drivers , It will also introduce the commonly used peripheral bus , Let everyone know what it is and why !
Third articles : At present, it is quite hot GUI LVGL Introduce , Mainly design LVGL7.1,LVGL8 Introduction to migration of , It will also introduce various components , After knowing the principle , Last , We will launch a configuration software to build our GUI, To improve our efficiency !
Fourth articles :ESP32- bluetooth , Familiar with my , We should all know , Even if I am engaged in the development of Bluetooth protocol stack , So this is our unique advantage , In this chapter , We will provide more than just knowledge of Bluetooth application methods , It will also apply the theory of combining Bluetooth underlying protocol stack , Let you completely get through the Bluetooth Ren Du pulse from top to bottom !
Chapter five :Wi-Fi Introduce , Familiar with my , I should know , We also made one sdio wifi Driving tutorial board , So in wifi We also have unique advantages in this regard , In this chapter , We also don't just offer Wi-Fi Knowledge of application , It will also be combined with the underlying theory , Make you right Wi-Fi There's a clear perception !
in addition , Our tutorial includes but is not limited to the above chapters , To give you a better navigation , The following information is particularly important , Please check in detail !!
------------------------------------------------------------------------------------------------------------------------------------------
Buy development boards ( Click on the I )
Document directory ( Click on the I )
Github Code warehouse ( Click on the I )
Bluetooth AC button group :539357317
WeChat official account ↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓
------------------------------------------------------------------------------------------------------------------------------------------
One . The concept of message queuing
Queue is also called message queue , It is a data structure commonly used for inter task communication , Queues can be between tasks 、 Interrupt and transfer information between tasks , It realizes that the task receives messages of variable length from other tasks or interrupts , The task can read messages from the queue , When the message in the queue is empty , The task of reading messages will be blocked , The user can also specify the blocked task time xTicksToWait, During this time , If the queue is empty , The task will remain blocked waiting for the queue data to be valid . When there are new messages in the queue , Blocked tasks will wake up and process new messages ; When the waiting time exceeds the specified blocking time , Even if there is no valid data in the queue , The task will also automatically change from blocking state to ready state . Message queuing is an asynchronous way of communication . Through Message Queuing service , A task or interrupt service routine can put one or more messages into a message queue . Again , One or more tasks can get messages from message queues . When multiple messages are sent to the message queue , Usually, the message that enters the message queue first is transmitted to the task first , in other words , What the task gets first is the message that first enters the message queue , That is, the first in, first out principle (FIFO), But it also supports the principle of last in, first out (LIFO).FreeRTOS The queue data structure is used to realize asynchronous communication of tasks , Has the following characteristics :
Message support first in first out queuing , Support asynchronous read-write mode .
Both read and write queues support timeout mechanism .
Message support LIFO queuing , Send message to team leader (LIFO).
Different lengths can be allowed ( No more than the maximum number of queue nodes ) Any type of message for .
A task can receive and send messages from any message queue .
Multiple tasks can receive and send messages from the same message queue .
When the queue is finished , You can delete by deleting the queue function .
Two . The working mechanism of message queue
When creating a message queue FreeRTOS A memory space will be allocated to the message queue first , The size of this memory is equal to the size of the message queue control block plus ( The product of the size of a single message space and the length of the message queue ), Then initialize the message queue , The message queue is empty .FreeRTOS The message queue control block of is composed of multiple elements , When the message queue is created , The system allocates the corresponding memory space for the control block , It is used to save some information of message queue, such as the storage location of message , The head pointer pcHead、 Tail pointer pcTail、 Message size uxItemSize And the length of the queue uxLength etc. . At the same time, each message queue is in the same continuous memory space as the message space , When the creation is successful , This memory is occupied , Only when the message queue is deleted , This memory will be released , When the creation is successful, the capacity of each message space and message queue has been allocated , You cannot change , Each message space can store no larger than the message size uxItemSize Any type of data , The length of the message queue is the total number of messages in the queue , This length can be specified when the message queue is created .
Task or interrupt service program can send message to message queue , When sending a message , If the queue is not full or overwriting is allowed ,FreeRTOS The message will be copied to the end of the message queue , otherwise , It will block according to the blocking timeout specified by the user , During this time , If the queue is never allowed to join , The task will remain blocked to wait for the queue to be allowed to queue . When other tasks read data from the waiting queue ( The queue is under ), The task will be automatically transferred from blocking state to ready state . When the waiting time exceeds the specified blocking time , Even if you're not allowed in the queue , The task will also automatically move from blocking state to ready state , At this time, the task or interrupt program sending the message will receive an error code errQUEUE_FULL.
The process of sending an emergency message is almost the same as sending a message , The only difference is , When sending an emergency message , The sending position is the message queue head, not the end , such , The receiver can receive the emergency message first , So as to process messages in time .
When a task tries to read a queue , It can specify a blocking timeout . During this time , If the queue is empty , The task will remain blocked waiting for the queue data to be valid . When another task or interrupt service program writes data to its waiting queue , The task will be automatically transferred from blocking state to ready state . When the waiting time exceeds the specified blocking time , Even if there is no valid data in the queue , The task will also automatically move from blocking state to ready state .
When message queuing is no longer in use , It should be removed to free up system resources , Once the operation is complete , Message queuing will
Permanently deleted .
The operation process of message queue is shown in Figure :
3、 ... and . Blocking mechanism of message queue
It's simple , because FreeRTOS It's done for us , We can use it directly , Each function that reads and writes to the message queue , All have this mechanism , I call it blocking mechanism . Suppose there is a task A When reading a queue ( That's what we call out of the team ), Found no message , Then the task is A Yes 3 A choice : The first option , Mission A Turn around and go , Since there are no messages in the queue , Then I won't wait , Do something else , Such subtasks A It will not enter the blocking state ; Second choice , Mission A Wait here , Maybe there will be messages in the queue later , Now the task A It's going to be blocked , Waiting for the news , And the task A The waiting time is defined by ourselves , Setting up 1000 A system clock beat tick The waiting for , Here 1000 individual tick Mission before arrival A It's all blocked , When blocking this time task A Wait for the message in the queue , So the task A It's going to change from blocking state to ready state , If this task A Higher priority than the currently running task , that , Mission A You get the message and run ; If 1000 individual tick It's all over. , There's no news in the queue , That mission A I don't want to wait , Wake up from blocking , Return an error code that did not wait for the message , And then continue with the mission A Other code for ; The third option , Mission A Death etc. , Don't leave until the news , Such subtasks A It's going to be blocked , Until the read queue message is complete .
When sending messages , To protect data , If and only if the queue is allowed to join , The sender can send the message successfully ; When there is no message space available in the queue , Indicates that the message queue is full , here , The system will block the task according to the blocking timeout specified by the user , If the queue joining operation cannot be completed within the specified timeout , The task sending the message or the interrupt service program will receive an error code errQUEUE_FULL, Then remove the blocking state ; Of course , Blocking status is allowed only when messages are sent in the task , Sending a message in an interrupt does not allow a message with a blocking mechanism , You need to call... To send a message in an interrupt API The function interface , Because the context in which the message is sent is in interrupt , No blocking is allowed .
Suppose there are multiple tasks blocked in a message queue , Then the blocked tasks will be sorted according to their priority , The task with higher priority will get the priority to access the queue .
Four . Common functions of message queue
The typical process of using the queue module is as follows :
Create a message queue .
Write queue operation .
Read queue operation .
To delete a queue .
1. Message queue creation function xQueueCreate()
xQueueCreate() Used to create a new queue and return the queue handle that can be used to access the queue . A queue handle is actually a pointer to the type of queue data structure .
A queue is a data structure , Used for data transfer between tasks . Every time a new queue is created, it needs to be assigned RAM , One part is used to store the status of the queue , be left over As the storage area of queue messages . Use xQueueCreate() Create a queue when , Dynamic memory allocation is used , therefore To use this function, you must FreeRTOSConfig.h Zhongba configSUPPORT_DYNAMIC_ALLOCATION Defined as 1 Enable , This is a macro to enable dynamic memory allocation , Usually , stay FreeRTOS in , Whenever you create a task , queue , Kernel objects such as semaphores and mutexes need to use dynamic memory allocation , So this macro defaults to FreeRTOS.h Enabled in header file ( Is defined as 1). If you want to use static memory , You can use xQueueCreateStatic() Function to create a queue . More formal parameters are required when using the static create message queue function to create a queue , The required memory is pre allocated during compilation , This method is rarely used . See the table for instructions
2. Message queue static creation function xQueueCreateStatic()
xQueueCreateStatic() Used to create a new queue and return the queue handle that can be used to access the queue . A queue handle is actually a pointer to the type of queue data structure .
A queue is a data structure , Used for data transfer between tasks . Every time a new queue is created, it needs to be assigned RAM , One Ministry branch use On save Store team Column Of shape state , Remnant Next Of do by team Column Of save Store District . send use xQueueCreateStatic() When creating a queue , Static memory allocation is used , So if you want to use this function, you must FreeRTOSConfig.h Zhongba configSUPPORT_STATIC_ALLOCATION Defined as 1 Enable . This is a macro to enable static memory allocation , The required memory is allocated when the program is compiled , Defined by the user , In fact, the creation process is similar to xQueueCreate() It's all the same , We won't go into details for the time being
3. Message queue delete function vQueueDelete()
The queue deletion function is directly deleted according to the message queue handle , After deletion, all information in the message queue will be recycled and emptied by the system , And you can't use this message queue again , But here's the thing , If a message queue is not created , That can't be deleted , Think about it , What is not created does not exist , How can it be deleted .xQueue yes vQueueDelete() The formal parameter of the function , Is a message queue handle , Indicates which queue to delete .
4. Send message function to message queue
Task or interrupt service program can send message to message queue , When sending a message , If the queue is not full or overwriting is allowed ,FreeRTOS The message will be copied to the end of the message queue , otherwise , It will block according to the blocking timeout specified by the user , During this time , If the queue is never allowed to join , The task will remain blocked to wait for the queue to be allowed to queue . When other tasks read data from the waiting queue ( The queue is under ), The task will automatically change from blocking state to ready state . When the waiting time of the task exceeds the specified blocking time , Even if you're not allowed in the queue , The task will also automatically move from blocking state to ready state , At this time, the task or interrupt program that sends the message will receive
An error code errQUEUE_FULL.
The process of sending an emergency message is almost the same as sending a message , The only difference is , When sending an emergency message , The sending position is the message queue head, not the end , such , The receiver can receive the emergency message first , So as to process messages in time .
In fact, there are several message queue sending functions , Are expanded using macro definitions , Some can only be called in the task , Some can only be called middle note. , See the following explanation for details .
4.1 xQueueSend() And xQueueSendToBack()
xQueueSend() Used to send a queue message to the end of the queue . Messages are queued as copies , Not in the form of a reference . This function can never be called in interrupt service program , Interrupt protection function must be used in interrupt xQueueSendFromISR() Instead of .xQueueSend() etc. Same as On xQueueSendToBack()
4.2 xQueueSendFromISR() And xQueueSendToBackFromISR()
xQueueSendFromISR() Is a macro , Macro expansion is to call the function xQueueGenericSendFromISR(). The macro is xQueueSend() Interrupt protected version of , Used to send a queue message to the end of the queue in the interrupt service program , Equivalent to xQueueSendToBackFromISR()
4.3 xQueueSendToFront()
xQueueSendToFron() yes One macro , macro exhibition open also yes transfer use function xQueueGenericSend() .xQueueSendToFront() Used to send a message to the queue leader . Messages are queued as copies , Not in the form of a reference . This function must not be called in the interrupt service program , Instead, you must use a with interrupt protection
xQueueSendToFrontFromISR () Instead of
4.4 xQueueSendToFrontFromISR()
xQueueSendToFrontFromISR() yes One individual macro , macro exhibition open yes transfer use function xQueueGenericSendFromISR(). The macro is xQueueSend ToFront() Interrupt protected version of , Used to send a message to the head of the message queue in the interrupt service program .
5. Read message function from message queue
5.1 xQueueReceive() And xQueuePeek()
xQueueReceive() yes One macro , macro exhibition open yes transfer use function xQueueGenericReceive() .xQueueReceive() Used to receive messages from a queue and remove them from the queue . The received message is in the form of copy , So we must provide a buffer with enough space . How much data can be copied to the buffer , This is set when the queue is created . This function must not be called in the interrupt service program , Instead, you must use a with interrupt protection xQueueReceiveFromISR () Instead of .
See here , Someone asked me what to do if I received the message and didn't want to delete it ? Actually , What you can think of ,FreeRTOS See also thought of , If you don't want to delete messages , Just call xQueuePeek () function . In fact, this function is similar to xQueueReceive() Functions are implemented in the same way , Even the use method is the same , It's just xQueuePeek() After receiving the message, the function will not delete the message in the message queue
5.2 xQueueReceiveFromISR() And xQueuePeekFromISR()
xQueueReceiveFromISR() yes xQueueReceive () Broken version of , It is used to receive a queue message in the interrupt service program and delete the message from the queue ;xQueuePeekFromISR() yes xQueuePeek() Broken version of , Used to receive messages from a queue in an interrupt , But it does not remove the message from the queue . To put it bluntly, these two functions can only be used to interrupt , There is no blocking mechanism , And it can be safely called in interrupt .
xQueuePeekFromISR Follow xQueueReceiveFromISR It works the same , Receiving a message from an interrupt , But the message will not be deleted from the message queue .
5、 ... and . Example of message queue
1. Code
#include <stdio.h>
#include <string.h>
#include <stdint.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "freertos/queue.h"
static QueueHandle_t message_queue;
typedef struct {
uint8_t *q_data;
uint16_t q_data_len;
} message_data_t;
void msg_queue_send_task(void *pvParameters)
{
printf("msg_queue_send_task\n");
while(1)
{
uint8_t data_len = rand() % 256;
message_data_t message_data;
if(!data_len)
data_len = 1;
message_data.q_data_len = data_len;
message_data.q_data = malloc(data_len);
if (message_data.q_data == NULL) {
printf("Malloc q_data_len failed!");
return;
}
printf("message data len = %d\n",data_len);
memset(message_data.q_data,data_len,data_len);
if (xQueueSend(message_queue, (void *)&message_data, ( TickType_t ) 0) != pdTRUE) {
printf("Failed to enqueue message_queue. Queue full.");
/* If data sent successfully, then free the pointer in `xQueueReceive'
* after processing it. Or else if enqueue in not successful, free it
* here. */
free(message_data.q_data);
}
vTaskDelay(1000 / portTICK_RATE_MS);
}
}
void app_main(void)
{
message_data_t message_data;
message_queue = xQueueCreate(15, sizeof(message_data_t));
if (message_queue == NULL) {
printf("Queue creation failed\n");
return;
}
xTaskCreate(&msg_queue_send_task, "msg_queue_send_task", 2048, NULL, 6, NULL);
while(1)
{
if (xQueueReceive(message_queue, &message_data, portMAX_DELAY) != pdPASS) {
printf("Queue receive error");
} else {
printf("message_data len:%d\n",message_data.q_data_len);
printf("message_data[0]=%d message_data[%d]=%d\n",message_data.q_data[0],
message_data.q_data_len-1,message_data.q_data[message_data.q_data_len-1]);
free(message_data.q_data);
}
}
}
I understand the whole code API The use of is super simple , To create a task, And then in task Every 1s Insert a data into the message queue , That's what the data looks like , The length of the data is random 1~255, then buffer All data in is equal to length , Then the data is received all the time in the main loop , And print it out
Reference Content : Wildfire FreeRTOS Application and implementation of kernel
版权声明
本文为[Wireless_ Link]所创,转载请带上原文链接,感谢
https://yzsam.com/2022/04/202204230144192800.html
边栏推荐
- Server 2019 the available memory of the server is half of the actual memory
- Batch multiple files into one hex
- Leetcode-阶乘函数后 K 个零
- What is an API interface?
- 什么是布尔类型?
- Quel est le fichier makefile?
- FL studio20.8最新中文版本安装下载图文教程
- JSP basic knowledge summary
- Question bank and online simulation examination for safety management personnel of hazardous chemical business units in 2022
- CDR2022首发全新版本性能介绍
猜你喜欢
Challenges often faced by client project management
什么是bgp服务器,有哪些优势?
[experience tutorial] Alipay balance automatically transferred to the balance of treasure how to set off, cancel Alipay balance automatically transferred to balance treasure?
批处理多个文件合成一个HEX
Virtual serial port function of j-link V9 using skills
搭建个人主页保姆级教程(二)
What business scenarios will the BGP server be used in?
搭建网站是用物理机还是云主机好?
Solve the problem when installing MySQL
揭秘被Arm编译器所隐藏的浮点运算
随机推荐
Ziguang micro financial report is outstanding. What does the triple digit growth of net profit in 2021 depend on
使用代理IP是需要注意什么?
2022第六季完美童模 IPA国民赛领跑元宇宙赛道
C语言中如何“指名道姓”的进行初始化
拨号服务器是什么,有什么用处?
W801 / w800 WiFi socket development (I) - UDP
The sixth season of 2022, the perfect children's model IPA national race leads the yuanuniverse track
Use yolov4 on colab
动态代理ip的测试步骤有哪些?
RuntimeError: The size of tensor a (4) must match the size of tensor b (3) at non-singleton dimensio
NPM yarn startup error [resolved]
npm——配置淘宝镜像
2022.4.10-----leetcode.804
.NET单元测试第一篇:常见.NET单元测试框架有哪些?
[tutorial] how to use GCC "zero assembly" for white whoring MDK
Jerry's factors that usually affect CPU performance test results are: [article]
Analyze the advantages and disadvantages of tunnel proxy IP.
FL studio20.8最新中文版本安装下载图文教程
数字藏品平台入驻 数字藏品平台开发 数字藏品SaaS平台
CC2541的仿真器CC Debugger使用教程