FreeRTOS系列|消息队列一

2023-05-16

消息队列一

1. 消息队列简介

消息队列可以在任务与任务、任务与中断之间传递消息,队列可以保存有限个具有确定长度的数据单元。队列可保存的最大单元数目被称为队列的长度,在队列创建时需要指定其长度和每个单元(队列项或消息)的大小。通常情况下,队列被作为FIFO(先进先出)使用,即数据由队尾写入,从队首读出。当然由队列首写入也是可能的

在这里插入图片描述
队列有如下特性:

  • 数据存储:往队列写入数据是通过字节拷贝把数据复制存储到队列中;从队列读出数据使得把队列中的数据拷贝删除
  • 可被多任务存取:队列不是属于某个特别指定的任务的,任何任务都可以向队列中发送消息,或者从队列中提取消息
  • 读队列时阻塞:当某个任务试图读一个队列时,其可以指定一个阻塞超时时间。在这段时间中,若队列为空,该任务将保持阻塞状态以等待队列数据有效。当其他任务或中断服务程序往其等待的队列中写入了数据,该任务将自动由阻塞态转移为就绪态。当等待的时间超过了指定的阻塞时间,即使队列中尚无有效数据,任务也会自动由阻塞态转移为就绪态
  • 写队列时阻塞:任务也可以在写队列时指定一个阻塞超时时间。这个时间是当被写队列已满时,任务进入阻塞态以等待队列空间有效的最长时间

队列读写过程如下图所示

在这里插入图片描述

队列的结构体 Queue_t 如下示,该结构体在queue.c文件中定义:

typedef struct QueueDefinition
{
	int8_t *pcHead;	/* 指向队列存储区开始地址 */
	int8_t *pcTail;	/* 指向队列存储区最后一个字节 */
	int8_t *pcWriteTo; /* 指向存储区中下一个空闲区域 */

	union
	{
		/* 当用作队列的时候指向最后一个出队的队列项首地址 */
		int8_t *pcReadFrom;	
		/* 当用作递归互斥量的时候用来记录递归互斥量被调用的次数 */		
		UBaseType_t uxRecursiveCallCount;
	} u;

	List_t xTasksWaitingToSend;		/* 等待发送任务列表 */
	List_t xTasksWaitingToReceive;	/* 等待接收任务列表 */

	volatile UBaseType_t uxMessagesWaiting;/* 队列中当前队列项数量(消息数) */
	UBaseType_t uxLength;			/* 创建队列时指定的队列长度 */
	UBaseType_t uxItemSize;			/* 创建队列时指定的每个队列项(消息)最大长度 */
	/* 当队列上锁后用来统计从队列中接收到(出队的)的队列项数量,若未上锁为queueUNLOCKED */
	volatile int8_t cRxLock;
	/* 当队列上锁后用来统计发送到(入队的)队列中的队列项数量,若未上锁为queueUNLOCKED */		
	volatile int8_t cTxLock;		

	#if( ( configSUPPORT_STATIC_ALLOCATION == 1 ) && ( configSUPPORT_DYNAMIC_ALLOCATION == 1 ) )
		uint8_t ucStaticallyAllocated;	/* 若使用静态存储,赋值为pdTURE */
	#endif

	#if ( configUSE_QUEUE_SETS == 1 )	/* 队列集相关宏 */
		struct QueueDefinition *pxQueueSetContainer;
	#endif

	#if ( configUSE_TRACE_FACILITY == 1 )	/* 跟踪调试相关宏 */
		UBaseType_t uxQueueNumber;
		uint8_t ucQueueType;
	#endif
} xQUEUE;
/* 老版本的FreeRTOS中队列可能会使用xQUEUE这个名字,新版本都用Queue_t */
typedef xQUEUE Queue_t;

2. 消息队列的函数应用

2.1 创建消息队列

在使用队列之前必须先创建队列,有动态和静态创建队列两种方法。以下四个函数原型在queue.c文件中定义

/********************动态创建队列**********************************************************************/
QueueHandle_t xQueueCreate(UBaseType_t uxQueueLength, //要创建的队列长度(队列的项目数)
						   UBaseType_t uxItemSize);	//队列项(消息)的长度(字节)
/********************动态创建队列**********************************************************************/
QueueHandle_t xQueueGenericCreate(const UBaseType_t uxQueueLength,//要创建的队列长度(队列的项目数)
								  const UBaseType_t uxItemSize,	//队列项(消息)的长度(字节)
								  const uint8_t ucQueueType); //队列类型
/********************静态创建队列**********************************************************************/
QueueHandle_t xQueueCreateStatic(UBaseType_t uxQueueLength,//要创建的队列长度(队列的项目数) 
								 UBaseType_t uxItemSize, //队列项(消息)的长度(字节)
								 uint8_t *pucQueueStorage, //指向队列项的存储区
								 StaticQueue_t *pxStaticQueue);//用来保存队列结构体
/********************静态创建队列**********************************************************************/
QueueHandle_t xQueueGenericCreateStatic(const UBaseType_t uxQueueLength,//要创建的队列长度(队列的项目数)
										const UBaseType_t uxItemSize,//队列项(消息)的长度(字节) 
										uint8_t *pucQueueStorage,//指向队列项的存储区 
										StaticQueue_t *pxStaticQueue,//用来保存队列结构体
										const uint8_t ucQueueType);//队列类型
/*****************************************************************************************************/
返回值:创建成功返回队列句柄;失败返回NULL

动态创建队列最终会调用xQueueGenericCreate()函数,下面来分析该源码(静态创建类似,不做分析)

QueueHandle_t xQueueGenericCreate(const UBaseType_t uxQueueLength, 
								  const UBaseType_t uxItemSize, 
								  const uint8_t ucQueueType ){
	Queue_t *pxNewQueue;
	size_t xQueueSizeInBytes;
	uint8_t *pucQueueStorage;
	if( uxItemSize == ( UBaseType_t ) 0 ){
		/* 如果队列项大小为0,就不需要存储区 */
		xQueueSizeInBytes = ( size_t ) 0; 
	}
	else{
		/* 最大队列项数量乘以单个队列项大小,就是消息存储区的大小 */
		xQueueSizeInBytes = ( size_t ) ( uxQueueLength * uxItemSize ); 
	}
	/* 给队列申请内存空间=队列结构体大小+消息存储区大小 */
	pxNewQueue = ( Queue_t * ) pvPortMalloc( sizeof( Queue_t ) + xQueueSizeInBytes );
	if( pxNewQueue != NULL ){	//内存申请成功
		/* 计算出消息存储区的首地址 */
		pucQueueStorage = ( ( uint8_t * ) pxNewQueue ) + sizeof( Queue_t );
		/* 初始化消息队列 */
		prvInitialiseNewQueue( uxQueueLength, uxItemSize, pucQueueStorage, ucQueueType, pxNewQueue );
	}
	return pxNewQueue;
}

队列初始化函数prvInitialiseNewQueue()源码分析

static void prvInitialiseNewQueue(const UBaseType_t uxQueueLength, 
								  const UBaseType_t uxItemSize, 
								  uint8_t *pucQueueStorage, 
								  const uint8_t ucQueueType, 
								  Queue_t *pxNewQueue){
	/* 防止编译器报错 */
	( void ) ucQueueType;
	if(uxItemSize == ( UBaseType_t ) 0){
		/* 若队列项长度为0,说明没有队列存储区 */
		pxNewQueue->pcHead = ( int8_t * ) pxNewQueue;
	}
	else{
		/* 把队列空间首地址指向队列项存储区首地址 */
		pxNewQueue->pcHead = ( int8_t * ) pucQueueStorage;
	}
	/* 初始化队列结构体相关成员变量 */
	pxNewQueue->uxLength = uxQueueLength;
	pxNewQueue->uxItemSize = uxItemSize;
	/* 队列复位 */
	( void ) xQueueGenericReset( pxNewQueue, pdTRUE );
}

队列复位函数xQueueGenericReset源码分析

BaseType_t xQueueGenericReset( QueueHandle_t xQueue, BaseType_t xNewQueue){
	Queue_t * const pxQueue = ( Queue_t * ) xQueue;
	//进入临界段,此时操作队列控制块,不允许被打断
	taskENTER_CRITICAL();
	{
		/* 初始化队列相关成员变量*/
		pxQueue->pcTail = pxQueue->pcHead + ( pxQueue->uxLength * pxQueue->uxItemSize );//头地址赋值
		pxQueue->uxMessagesWaiting = ( UBaseType_t ) 0U;//等待处理的消息个数为0
		pxQueue->pcWriteTo = pxQueue->pcHead;//写入指针赋值为队列头指针
		pxQueue->u.pcReadFrom = pxQueue->pcHead + ( ( pxQueue->uxLength - ( UBaseType_t ) 1U ) * pxQueue->uxItemSize );//读出指针写入最后一个可用消息
		pxQueue->cRxLock = queueUNLOCKED;//赋值队列锁为解锁状态
		pxQueue->cTxLock = queueUNLOCKED;//赋值队列锁为解锁状态
		/* 判断是否为新建队列,若不是还需要其他处理 */
		if( xNewQueue == pdFALSE ){
			/* 判断发送等待列表里面是否有任务 */
			if(listLIST_IS_EMPTY(&(pxQueue->xTasksWaitingToSend))==pdFALSE){
				/* 移除事件列表中的任务 */
				if(xTaskRemoveFromEventList(&(pxQueue->xTasksWaitingToSend))!=pdFALSE){
					/* 进行上下文切 */
					queueYIELD_IF_USING_PREEMPTION();
				}
				else{
					mtCOVERAGE_TEST_MARKER();
				}
			}
			else{
				mtCOVERAGE_TEST_MARKER();
			}
		}
		else{ 
			/* 若是新建队列,则直接初始化发送和接受列表项 */
			vListInitialise( &( pxQueue->xTasksWaitingToSend ) );
			vListInitialise( &( pxQueue->xTasksWaitingToReceive ) );
		}
	}
	taskEXIT_CRITICAL();
	return pdPASS;
}
2.2 向队列发送消息
  • 任务级入队函数
/**********发送消息到队尾********************************************************************************/
BaseType_t xQueueSend(QueueHandle_t xQueue,//队列句柄,指明向哪个队列发送数据
					  const void *pvItemToQueue,//指向要发送的消息
					  TickType_t xTicksToWait);//队列满时,任务进入阻塞态等待队列空闲的最大时间(阻塞时间)
/**********发送消息到队尾********************************************************************************/
BaseType_t xQueueSendToBack(QueueHandle_t xQueue,//队列句柄,指明向哪个队列发送数据
					  const void *pvItemToQueue,//指向要发送的消息
					  TickType_t xTicksToWait);//阻塞时间
/**********发送消息到队头********************************************************************************/
BaseType_t xQueueSendToFront(QueueHandle_t xQueue,//队列句柄,指明向哪个队列发送数据
					  const void *pvItemToQueue,//指向要发送的消息
					  TickType_t xTicksToWait);//阻塞时间
/**********发送消息到队列(带覆写功能,即队列满了以后会覆写掉旧的数据)*************************************/
BaseType_t xQueueOverwrite(QueueHandle_t xQueue,//队列句柄,指明向哪个队列发送数据
					  const void *pvItemToQueue);//指向要发送的消息
/**********任务级通用入队函数***************************************************************************/
BaseType_t xQueueGenericSend(QueueHandle_t xQueue, //队列句柄,指明向哪个队列发送数据
							 const void * const pvItemToQueue, //指向要发送的消息
							 TickType_t xTicksToWait,//阻塞时间
							 const BaseType_t xCopyPosition);//入队方式(后向/前向/覆写)
/******************************************************************************************************/
返回值:发送消息成功,返回pdPASS;队列满消息发送失败,返回errQUEUE_FULL

任务级入队函数最终都是调用xQueueGenericSend()函数,下面来分析该函数源码

BaseType_t xQueueGenericSend(QueueHandle_t xQueue, 
							 const void * const pvItemToQueue, 
							 TickType_t xTicksToWait, 
							 const BaseType_t xCopyPosition){
	BaseType_t xEntryTimeSet = pdFALSE, xYieldRequired;
	TimeOut_t xTimeOut;
	Queue_t * const pxQueue = ( Queue_t * ) xQueue;
	/* 使用for死循环,是为了快速的处理消息拷贝 */
	for( ;; ){		
		taskENTER_CRITICAL(); //进入了临界段
		{
			/* 判断消息队列是否满了以及是否允许覆盖入队,任一条件成立都执行入队操作	*/
			if((pxQueue->uxMessagesWaiting < pxQueue->uxLength)||(xCopyPosition == queueOVERWRITE))			{
				/* 拷贝数据到队列操作空间内 */
				xYieldRequired = prvCopyDataToQueue(pxQueue, pvItemToQueue, xCopyPosition);
				#if ( configUSE_QUEUE_SETS == 1 )
				{ /*......省略掉与队列集相关代码......*/	}
				#else /* configUSE_QUEUE_SETS */
				{
					/* 判断等待接收的列表是否为空. */
					if(listLIST_IS_EMPTY(&(pxQueue->xTasksWaitingToReceive)) == pdFALSE){
						/* 若不为空,表示有任务由于请求消息而阻塞,则改变阻塞态为就绪态. */
						if(xTaskRemoveFromEventList(&(pxQueue->xTasksWaitingToReceive)) != pdFALSE){
							/* 进行上下文切换*/
							queueYIELD_IF_USING_PREEMPTION();
						}
						else{
							mtCOVERAGE_TEST_MARKER();
						}
					}
					else if( xYieldRequired != pdFALSE){
						/* 再次进行上下文切换 */
						queueYIELD_IF_USING_PREEMPTION();
					}
					else{
						mtCOVERAGE_TEST_MARKER();
					}
				}
				#endif /* configUSE_QUEUE_SETS */
				taskEXIT_CRITICAL();	//退出临界段
				return pdPASS;			//返回pdPASS,标记入队成功
			}
			else{
				/* 队列满了不允许入队时,先判断是否需要阻塞 */
				if( xTicksToWait == ( TickType_t ) 0 ){
					/* 为0表示,表示没有阻塞时间 */
					taskEXIT_CRITICAL();//退出临界段
					traceQUEUE_SEND_FAILED( pxQueue );
					return errQUEUE_FULL;//返回队列已满
				}
				else if( xEntryTimeSet == pdFALSE ){
					/* 若有阻塞时间,就初始化时间结构体*/
					vTaskSetTimeOutState( &xTimeOut );
					xEntryTimeSet = pdTRUE;
				}
				else{
					/* 时间结构体已经初始化过了 */
					mtCOVERAGE_TEST_MARKER();
				}
			}
		}		
		taskEXIT_CRITICAL();//退出临界段		
		/* 执行到这里说明当前队列已满,并且设置了不为0的阻塞时间 */
		vTaskSuspendAll();//挂起任务调度器
		prvLockQueue( pxQueue );//给队列上锁
		/* 判断阻塞时间是否超时了 */
		if( xTaskCheckForTimeOut( &xTimeOut, &xTicksToWait ) == pdFALSE){
			/* 若未超时,则判断队列是否还是满的 */
			if( prvIsQueueFull( pxQueue ) != pdFALSE ){
				/* 若此时队仍满且未超时,则把当前任务添加到等待发送的事件列表和延时列表中去*/
				traceBLOCKING_ON_QUEUE_SEND( pxQueue );
				vTaskPlaceOnEventList( &( pxQueue->xTasksWaitingToSend ), xTicksToWait );	
				prvUnlockQueue( pxQueue );//解锁队列
				if( xTaskResumeAll() == pdFALSE ){//恢复任务调度器
					portYIELD_WITHIN_API();//进行上下文切换
				}
			}
			else{
				/* 若此时队列未满,但未超时,则重新进行入队操作 */
				prvUnlockQueue( pxQueue );
				( void ) xTaskResumeAll();
			}
		}
		else{
			/* 若已超时,则解锁队列,恢复任务调度器 */
			prvUnlockQueue( pxQueue );
			( void ) xTaskResumeAll();
			traceQUEUE_SEND_FAILED( pxQueue );
			return errQUEUE_FULL;//返回队列已满
		}
	}
}
  • 中断级入队函数
/**********发送消息到队尾*****************************************************************************/
BaseType_t xQueueSendFromISR(QueueHandle_t xQueue,//队列句柄,指明向哪个队列发送数据 
							 const void * pvItemToQueue,//指向要发送的消息 
							 BaseType_t * pxHigherPriorityTaskWoken);//标记退出此函数后是否进行任务切换
/**********发送消息到队尾****************************************************************************/
BaseType_t xQueueSendToBackFromISR(QueueHandle_t xQueue, 
							 	   const void * pvItemToQueue, 
							 	   BaseType_t * pxHigherPriorityTaskWoken);
/**********发送消息到队头****************************************************************************/
BaseType_t xQueueSendToFrontFromISR(QueueHandle_t xQueue, 
							 	    const void * pvItemToQueue, 
							 		BaseType_t * pxHigherPriorityTaskWoken);
/**********发送消息到队列(带覆写功能,即队列满了以后会覆写掉旧的数据)********************************/
BaseType_t xQueueOverwriteFromISR(QueueHandle_t xQueue, 
							 	  const void * pvItemToQueue, 
							 	  BaseType_t * pxHigherPriorityTaskWoken);
/**********中断级通用入队函数***********************************************************************/
BaseType_t xQueueGenericSendFromISR(QueueHandle_t xQueue, 
									const void * pvItemToQueue, 
									BaseType_t * pxHigherPriorityTaskWoken, 
									BaseType_t xCopyPosition);//入队方式(后向/前向/覆写)
/*************************************************************************************************/
返回值:发送消息成功,返回pdPASS;队列满消息发送失败,返回errQUEUE_FULL

中断级入队函数最终都是调用xQueueGenericSendFromISR()函数,下面来分析该函数源码

BaseType_t xQueueGenericSendFromISR(QueueHandle_t xQueue, 
									const void * const pvItemToQueue, 
									BaseType_t * const pxHigherPriorityTaskWoken, 
									const BaseType_t xCopyPosition){
	BaseType_t xReturn;
	UBaseType_t uxSavedInterruptStatus;
	Queue_t * const pxQueue = ( Queue_t * ) xQueue;
	/* 带返回值的关闭中断,需要保存上次关闭中断的状态值,恢复时候写入 */
	uxSavedInterruptStatus = portSET_INTERRUPT_MASK_FROM_ISR();
	{
		/* 判断消息队列是否满了以及是否允许覆盖入队,任一条件成立都执行入队操作	*/
		if( ( pxQueue->uxMessagesWaiting < pxQueue->uxLength ) || ( xCopyPosition == queueOVERWRITE ) )
		{
			/* 获取队列发送锁的状态值 */
			const int8_t cTxLock = pxQueue->cTxLock;
			/* 拷贝数据到队列操作空间内 */
			( void ) prvCopyDataToQueue( pxQueue, pvItemToQueue, xCopyPosition );
			/* 判断队列是否上锁 */
			if( cTxLock == queueUNLOCKED ){	//若队列未上锁
				#if ( configUSE_QUEUE_SETS == 1 )
				{/*......省略掉与队列集相关代码......*/}
				#else /* configUSE_QUEUE_SETS */
				{
					/* 判断等待接收的列表是否为空. */
					if(listLIST_IS_EMPTY(&(pxQueue->xTasksWaitingToReceive) == pdFALSE){
						/* 若不为空,表示有任务由于请求消息而阻塞,则改变阻塞态为就绪态. */
						if(xTaskRemoveFromEventList( &( pxQueue->xTasksWaitingToReceive))!= pdFALSE){
							/* 若上一步变成就绪态的任务优先级比当前任务高,则标记为pdTRUE,表示要进行任务切换 */
							if( pxHigherPriorityTaskWoken != NULL ){
								*pxHigherPriorityTaskWoken = pdTRUE;
							}
							else{
								mtCOVERAGE_TEST_MARKER();
							}
						}
						else{
							mtCOVERAGE_TEST_MARKER();
						}
					}
					else{
						mtCOVERAGE_TEST_MARKER();
					}
				}
				#endif /* configUSE_QUEUE_SETS */
			}
			else{ //若队列已经上锁
				/* 发送锁加一,表示进行了一次入队操作 */
				pxQueue->cTxLock = ( int8_t ) ( cTxLock + 1 );
			}
			xReturn = pdPASS;//返回pdPASS,表示入队完成
		}
		else{	//若队列满		
			traceQUEUE_SEND_FROM_ISR_FAILED( pxQueue );
			xReturn = errQUEUE_FULL;//返回errQUEUE_FULL,表示队列满
		}
	}
	/* 开启中断,保存上次状态值 */
	portCLEAR_INTERRUPT_MASK_FROM_ISR( uxSavedInterruptStatus );
	return xReturn;
}
2.3 队列上锁和解锁
  • 队列上锁 prvLockQueue():本质上是一个宏,定义如下
//就是将队列中的成员变量cRxLock和cTxLock设置为queueLOCKED_UNMODIFIED
#define prvLockQueue( pxQueue )								\
	taskENTER_CRITICAL();									\
	{														\
		if( ( pxQueue )->cRxLock == queueUNLOCKED )			\
		{													\
			( pxQueue )->cRxLock = queueLOCKED_UNMODIFIED;	\
		}													\
		if( ( pxQueue )->cTxLock == queueUNLOCKED )			\
		{													\
			( pxQueue )->cTxLock = queueLOCKED_UNMODIFIED;	\
		}													\
	}														\
	taskEXIT_CRITICAL()
  • 队列解锁
//上锁计数器cRxLock和cTxLock记录了队列上锁期间,入队或出队的数量
static void prvUnlockQueue( Queue_t * const pxQueue ){
	/* 进入临界段 */
	taskENTER_CRITICAL();
	{		
		int8_t cTxLock = pxQueue->cTxLock;//获取发送锁的状态值
		/* 遍历直到解锁为止,队列上锁后向队列发送消息成功,入队计数器cTxLock会加一 */
		while( cTxLock > queueLOCKED_UNMODIFIED ){
			#if ( configUSE_QUEUE_SETS == 1 )
			{/*......省略掉与队列集相关代码......*/}
			#else /* configUSE_QUEUE_SETS */
			{
				/* 判断等待接收的列表是否为空. */
				if( listLIST_IS_EMPTY( &( pxQueue->xTasksWaitingToReceive ) ) == pdFALSE ){
					/* 若不为空,将任务从列表中删除 */
					if(xTaskRemoveFromEventList(&(pxQueue->xTasksWaitingToReceive)) != pdFALSE){
						/* 若移除的任务优先级比当前任务高,则进行任务切换 */
						vTaskMissedYield();
					}
					else{
						mtCOVERAGE_TEST_MARKER();
					}
				}
				else{
					break;
				}
			}
			#endif /* configUSE_QUEUE_SETS */
			--cTxLock;//每处理完一条,队列发送锁就减一,直至处理完成
		}		
		pxQueue->cTxLock = queueUNLOCKED;//处理完成后,解除发送锁
	}	
	taskEXIT_CRITICAL();//退出临界段
	/* 接下来处理cRxLock,处理过程与cTxLock类似 */
	taskENTER_CRITICAL();
	{
		int8_t cRxLock = pxQueue->cRxLock;
		while( cRxLock > queueLOCKED_UNMODIFIED ){
			if(listLIST_IS_EMPTY(&(pxQueue->xTasksWaitingToSend)) == pdFALSE){
				if(xTaskRemoveFromEventList(&(pxQueue->xTasksWaitingToSend)) != pdFALSE){
					vTaskMissedYield();
				}
				else{
					mtCOVERAGE_TEST_MARKER();
				}
				--cRxLock;
			}
			else{
				break;
			}
		}
		pxQueue->cRxLock = queueUNLOCKED;
	}
	taskEXIT_CRITICAL();
}
2.4 从队列读取消息
  • 任务级出队函数
/**********从队列中读取队列项,读取完后删除队列项**************************************************/
BaseType_t xQueueReceive(QueueHandle_t xQueue,//队列句柄 
						 void * const pvBuffer,//保存数据的缓冲区
						 TickType_t xTicksToWait);//阻塞时间,表示队列空时进入阻塞态等待数据的最大时间
/**********从队列中读取队列项,读取完后不删除队列项************************************************/
BaseType_t xQueuePeek(QueueHandle_t xQueue, 
					  void * const pvBuffer, 
					  TickType_t xTicksToWait);
/**********从队列中读取队列项通用函数************************************************************/
BaseType_t xQueueGenericReceive(QueueHandle_t xQueue, 
								void * const pvBuffer, 
								TickType_t xTicksToWait, 
								const BaseType_t xJustPeeking);//标记读取成功后是否需要删除队列项
/**********************************************************************************************/
返回值:读取数据成功,返回pdTRUE;读取失败,返回pdFALSE

任务级出队函数最终都是调用xQueueGenericReceive()函数,出队与入队的源码流程差不多,在此不再赘述

  • 中断级出队函数
/**********从队列中读取队列项,读取完后删除队列项*******************************************/
BaseType_t xQueueReceiveFromISR(QueueHandle_t xQueue,//队列句柄 
						 	    void * const pvBuffer,//保存数据的缓冲区
						 		BaseType_t * pxTaskWoken);//标记退出函数后是否进行任务切换
/**********从队列中读取队列项,读取后不删除队列项*******************************************/
BaseType_t xQueuePeekFromISR(QueueHandle_t xQueue,//队列句柄 
						 	 void * const pvBuffer,//保存数据的缓冲区);
返回值:读取数据成功,返回pdTRUE;读取失败,返回pdFALSE

关注我的公众号,在公众号里发如下消息,即可获取相应的工程源代码:

FreeRTOS消息队列实例

在这里插入图片描述

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

FreeRTOS系列|消息队列一 的相关文章

  • FreeRTOS快速上手

    FreeRTOS使用 一 源码下载和移植文件提取 1 1 源码下载 在网站https sourceforge net projects freertos 可以找到freertos最新的源码 1 2 移植文件提取 根据第一步 我们会得到一个f
  • FreeRTOS学习(八) 延时函数

    声明及感谢 跟随正点原子资料学习 在此作为学习的记录和总结 环境 keil stm32f103 FreeRTOS延时函数有两个 分别是 vTaskDelay vTaskDelayUntil 1 vTaskDelay 任务相对延时 函数原型
  • FreeRTOS系列

    1 RTOS简介 RTOS全称为 Real Time Operation System 即实时操作系统 RTOS强调的是实时性 又分为硬实时和软实时 硬实时要求在规定的时间内必须完成操作 不允许超时 而软实时里对处理过程超时的要求则没有很严
  • 【FreeRTOS】队列的使用

    作者主页 凉开水白菜 作者简介 共同学习 互相监督 热于分享 多加讨论 一起进步 专栏资料 https pan baidu com s 1nc1rfyLiMyw6ZhxiZ1Cumg pwd free 点赞 收藏 再看 养成习惯 订阅的粉丝
  • FreeRTOS+CubeMX系列第一篇——初识FreeRTOS

    文章目录 一 关于FreeRTOS 二 FreeRTOS的特点 三 如何在CubeMX上配置FreeRTOS 四 FreeRTOS文档资料 五 同系列博客 一 关于FreeRTOS 1 什么是FreeRTOS FreeRTOS是一个迷你的实
  • FreeRTOS临界区

    FreeRTOS临界区是指那些必须完整运行 不能被打断的代码段 比如有的外设的初始化需要严格的时序 初始化过程中不能被打断 FreeRTOS 在进入临界区代码的时候需要关闭中断 当处理完临界区代码以后再打开中断 FreeRTOS 系统本身就
  • FreeRTOS记录(九、一个裸机工程转FreeRTOS的实例)

    记录一下一个实际项目由裸机程序改成FreeRTOS 以前产品的平台还是C8051单片机上面的程序 硬件平台改成了STM32L051 同时使用STM32CubeMX生成的工程 使用FreeRTOS系统 EEPROM数据存储读取函数修改更新 2
  • FreeRTOS_中断

    传送门 博客汇总帖 传送门 Cortex M3 中断 异常 传送门 Cortex M3笔记 基础 笔记内容参考 正点原子的FreeRTOS开发手册 cortex m3权威指南 Cortex M3和Cortex M4权威指南等 文中stm32
  • Error: L6218E: Undefined symbol vApplicationGetIdleTaskMemory (referred from tasks.o).

    我用的是F103ZET6的板子 移植成功后 编译出现两个错误是关于stm32f10x it c 里 void SVC Handler void void PendSV Handler void 两个函数的占用问题 随后编译出现以下两个问题
  • 基于HAL库的FREERTOS-----------三.队列

    一 队列简介 在实际的应用中 常常会遇到一个任务或者中断服务需要和另外一个任务进行 沟通交流 这个 沟通交流 的过程其实就是消息传递的过程 在没有操作系统的时候两个应用程序进行消息传递一般使用全局变量的方式 但是如果在使用操作系统的应用中用
  • FreeRTOS基础五:软件定时器

    软件定时器简介 软件定时器的作用 在指定的时间到来时执行指定的函数 或者以某个频率周期性地执行某个函数 被执行的函数叫做软件定时器回调函数 软件定时器由FreeRTOS内核实现 不需要硬件支持 软件定时器只有在软件定时器回调函数被调用时才需
  • FreeRTOS,串口中断接收中使用xQueueOverwriteFromISR()函数,程序卡死在configASSERT

    原因 UART的中断优先级设置的太高 高于了configMAX SYSCALL INTERRUPT PRIORITY宏定义的安全中断等级 UART的中断等级小于等于宏定义的优先等级即可
  • stm32f103zet6移植标准库的sdio驱动

    sdio移植 st官网给的标准库有给一个用于st出的评估板的sdio外设实现 但一是文件结构有点复杂 二是相比于国内正点原子和野火的板子也有点不同 因此还是需要移植下才能使用 当然也可以直接使用正点原子或野火提供的实例 但为了熟悉下sdio
  • FreeRTOS之软件定时器

    FreeRTOS之软件定时器 声明 本人按照正点原子的FreeRTOS例程进行学习的 欢迎各位大佬指责和批评 谢谢 include sys h include delay h include usart h include led h in
  • FreeRTOS死机原因

    1 中断回调函数中没有使用中断级API xxFromISR 函数 xSemaphoreGiveFromISR uart busy HighterTask 正确 xSemaphoreGive uart busy 错误 2 比configMAX
  • FreeRTOS轻量级同步--任务通知

    1 简介 在FreeRTOS的配置参数中的configUSE TASK NOTIFICATIONS宏打开 一般RTOS会默认打开 如图1所示 图1 notify宏开关 RTOS在创建任务时 会创建一个32位的通知值ulNotifiedVal
  • FreeRTOS多任务调度器基础

    Cortex M4中SysTick调度器核心 Cortex M4中的中断管理 Cortex M4中影子栈指针 Cortex M4中SVC和PendSV异常 1 Cortex M4中SysTick调度器核心 systick每一次中断都会触发内
  • 使用 GCC 编译器的 ARM 内核的堆栈回溯(当存在 MSP 到 PSP 切换时)

    核心 ARM Cortex M4 编译器 GCC 5 3 0 ARM EABI 操作系统 免费 RTOS 我正在使用 gcc 库函数 Unwind Reason Code Unwind Backtrace Unwind Trace Fn v
  • 如何更改 FreeRTOS 中任务的最大可用堆大小?

    我通过以下方式在任务中创建元素列表 l dllist pvPortMalloc sizeof dllist dlllist 有 32 字节大 我的嵌入式系统有 60kB SRAM 所以我希望系统可以轻松处理我的 200 个元素列表 我发现在
  • 如何将 void* 转换为函数指针?

    我在 FreeRTOS 中使用 xTaskCreate 其第四个参数 void const 是传递给新线程调用的函数的参数 void connect to foo void const task params void on connect

随机推荐

  • 如何在keil官网上下载 STM32包 pack

    https www 360kuai com pc 990a8ec633cf2109c cota 61 4 amp tj url 61 xz amp sign 61 360 57c3bbd1 amp refer scene 61 so 1
  • Makefile学习笔记系列1:一个简单的Makefile

    开启Makefile系列学习前 xff0c 先来个简单的 xff0c 没有子目录结构的例子 xff0c 只有一个makefile文件 目录结构 xff1a Makefile代码 xff1a XX 61 g 43 43 CFLAGS 61 g
  • PID控制算法的C语言实现

    PID控制算法的C语言实现一 PID算法原理 最近两天在考虑一般控制算法的C语言实现问题 xff0c 发现网络上尚没有一套完整的比较体系的讲解 于是总结了几天 xff0c 整理一套思路分享给大家 在工业应用中PID及其衍生算法是应用最广泛的
  • pageCache

    写在前面 在开始正式的讨论前 xff0c 我先抛出几个问题 xff1a 谈到磁盘时 xff0c 常说的HDD磁盘和SSD磁盘最大的区别是什么 xff1f 这些差异会影响我们的系统设计吗 xff1f 单线程写文件有点慢 xff0c 那多开几个
  • 基于Python高光谱遥感影像处理实例

    前言 在写波段配准相关代码时经常需要用到tif影像的波段合成和分解 xff0c 虽然可以用ENVI才处理 xff0c 但是每次都要打开再设置一些参数有些麻烦 xff0c 所以本着 独立自主 自力更生 的原则就写了些脚本来处理这个需求 又写了
  • 基于SIFT的图像Matlab拼接教程

    前言 图像拼接技术 xff0c 将普通图像或视频图像进行无缝拼接 xff0c 得到超宽视角甚至360度的全景图 xff0c 这样就可以用普通数码相机实现场面宏大的景物拍摄 利用计算机进行匹配 xff0c 将多幅具有重叠关系的图像拼合成为一幅
  • PERSIANN 降雨数据使用教程

    一 前言 PERSIANN xff0c 使用人工神经网络从遥感信息中估算降水 xff0c 是一种基于卫星的降水检索算法 xff0c 可提供近乎实时的降雨信息 该算法使用来自全球地球同步卫星的红外 IR 卫星数据作为降水信息的主要来源 红外图
  • 基于Pyqt5快速构建应用程序详细教程

    一 介绍 图形用户界面 xff0c 更广为人知的名称是 GUI xff0c 是当今大多数个人计算机的一个特征 它为不同计算技能水平的用户提供了直观的体验 尽管 GUI 应用程序可能会使用更多资源 xff0c 但由于其点击式特性 xff0c
  • 基于Python的PROSAIL模型介绍以及使用

    1 介绍 PROSAIL是两种模型耦合得到的 SAIL是冠层尺度的辐射传输模型 xff0c 把冠层假设成是连续的且具有给定几何形状和密度的水平均匀分布的介质层 xff0c 从而模拟入射辐射与均匀介质之间的相互作用 xff0c 具体还是挺复杂
  • 可能是最全的FreeRTOS源码分析及应用开发系列

    可能是最全的FreeRTOS源码分析及应用开发系列 FreeRTOS 是一个可裁剪的小型且免费的 RTOS 系统 xff0c 尺寸非常小 xff0c 可运行于微控制器上 其特点包括 xff1a 内核支持抢占式 xff0c 合作式和时间片调度
  • 关于VS中LNK1120与errorLNK2019问题

    最近遇到了该问题 xff0c 再查找了一些资料后 xff0c 发现了针对自己问题的解决方法 xff0c 贴出来让大家一起学习一下 其实如果这两个问题同时出现 xff0c 很可能不是链接库缺了lib xff0c 而是编译中添加的源没有被实例化
  • PCL—低层次视觉—点云分割(基于凹凸性)

    转自 xff1a http www cnblogs com ironstark p 5027269 html PCL 低层次视觉 点云分割 xff08 基于凹凸性 xff09 1 图像分割的两条思路 场景分割时机器视觉中的重要任务 xff0
  • 【ENVI入门系列】13.分类后处理

    原文地址 xff1a ENVI入门系列 13 分类后处理 作者 xff1a ENVI IDL中国 版权声明 xff1a 本教程涉及到的数据提供仅练习使用 xff0c 禁止用于商业用途 目录 分类后处理 1 概述 2 分类后处理 2 1 小斑
  • ENVI神经网络工具参数和使用方法

    原文地址 xff1a ENVI神经网络工具参数和使用方法 作者 xff1a pengheligis xff08 1 xff09 Activation xff1a 选择活化函数 对数 xff08 Logistic xff09 和双曲线 xff
  • 详解使用pscp命令Linux文件上传与下载

    一 上传 2 开始 运行 cmd进入到 dos模式输入以下命令 以下是代码片段 xff1a pscp D java apache tomcat 5 5 27 webapps szfdc rardev 64 192 168 68 249 ho
  • java学习总结及心得体会

    前言 xff1a 哈哈 xff0c 今天是2015年 xff18 月 xff12 号 xff0c 星期日 xff0c 今天是收货的一天 xff0c 很开心 xff0c 很快乐 到底发生了什么呢 xff0c 容我慢慢来 世界很大 xff0c
  • 二进制的表白

    没能提起勇气对她进行表白 xff0c 只能寄托于0 1代码记录下对你的喜欢 01000101 01110110 01100101 01101110 00100001 01001001 00100000 01101100 01101111 0
  • C语言strcat()函数:连接字符串

    头文件 xff1a include lt string h gt strcat 函数用来连接字符串 xff0c 其原型为 xff1a char strcat char dest const char src 参数 dest 为目的字符串指针
  • linux 下TCP通信例程

    TCP server span class token macro property span class token directive hash span span class token directive keyword inclu
  • FreeRTOS系列|消息队列一

    消息队列一 1 消息队列简介 消息队列可以在任务与任务 任务与中断之间传递消息 xff0c 队列可以保存有限个具有确定长度的数据单元 队列可保存的最大单元数目被称为队列的长度 xff0c 在队列创建时需要指定其长度和每个单元 xff08 队