赞
踩
本文目标:FreeRTOS学习第10篇–队列使用示例
按照本文的描述,可以进行简单的使用队列。
本文实验条件:拥有C语言基础,装有编译和集成的开发环境,比如:Keil uVision5
在本次实验中,继续沿用上一篇的工程文件,将输入通过遥控器获取的数据改用队列的方式进行实现。
函数原型
QueueHandle_t xQueueCreate( UBaseType_t uxQueueLength, UBaseType_t uxItemSize );
函数描述:
函数 xQueueCreate 用于创建队列。
第 1 个参数uxQueueLength是队列支持的消息个数,最多能存放多少个数据(item)
第 2 个参数uxItemSize是每个消息的大小,单位字节。
返回值:非 0:成功,返回句柄,以后使用句柄来操作队列NULL:失败,因为内存不足
函数原型
BaseType_t xQueueSend(
QueueHandle_t xQueue, /* 消息队列句柄 */
const void * pvItemToQueue, /* 要传递数据地址 */
TickType_t xTicksToWait /* 等待消息队列有空间的最大等待时间*/
);
函数描述:
第 1 个参数xQueue是消息队列句柄。
第 2 个参数pvItemToQueue要传递数据地址,每次发送都是将消息队列创建函数 xQueueCreate 所指定的单个消息大
小复制到消息队列空间中。
第 3 个参数xTicksToWait是当消息队列已经满时,等待消息队列有空间时的最大等待时间,单位系统时钟节拍。如果被设为 0,无法写入数据时函数会立刻返回;如果被设为 portMAX_DELAY,则会一直阻塞直到有空间可写
返回值,如果消息成功发送返回 pdPASS,失败返回 errQUEUE_FULL。
使用这个函数要注意的点:
FreeRTOS 的消息传递是数据的复制,而不是传递的数据地址。
此函数是用于任务代码中调用的,故不可以在中断服务程序中调用此函数,中断服务程序中使用的是xQueueSendFromISR。
如果消息队列已经满且第三个参数为 0,那么此函数会立即返回。
如果用户将 FreeRTOSConfig.h 文件中的宏定义 INCLUDE_vTaskSuspend 配置为 1 且第三个参数配置为 portMAX_DELAY,那么此发送函数会永久等待直到消息队列有空间可以使用。
消息队列还有两个函数 xQueueSendToBack 和 xQueueSendToFront,函数 xQueueSendToBack实现的是 FIFO 方式的存取,函数 xQueueSendToFront 实现的是 LIFO 方式的读写。我们这里说的函数 xQueueSend 等效于 xQueueSendToBack,即实现的是 FIFO 方式的存取。
函数原型
BaseType_t xQueueReceive(
QueueHandle_t xQueue, /* 消息队列句柄 */
void *pvBuffer, /* 接收消息队列数据的缓冲地址 */
TickType_t xTicksToWait /* 等待消息队列有数据的最大等待时间 */
);
函数描述:
第 1 个参数xQueue是消息队列句柄。
第 2 个参数pvBuffer要传递数据地址,bufer 指针,队列的数据会被复制到这个 buffer复制多大的数据?在创建队列时已经指定了数据大小。缓冲区空间要大于等于消息队列创建函数 xQueueCreate 所指定的单个消息大小,否则取出的数据无法全部存储到缓冲区,从而造成内存溢出。
第 3 个参数xTicksToWait是当消息队列为空时,等待消息队列有数据的最大等待时间,单位系统时钟节拍。
在本次工程中,按键获取数据的方式来自中断,在中断进行写队列,调用的如下的代码进行写队列:
// 在中断的代码进行调用
/* 写队列 */
idata.dev = datas[0];
idata.val = datas[2];
xQueueSendFromISR(g_xQueuePlatform, &idata, NULL);
而在任务中使用如下的代码进行读队列
xQueueReceive(g_xQueuePlatform, &idata, portMAX_DELAY);
在game1_task游戏中的代码片段
void game1_task(void *params) { uint8_t dev, data, last_data; struct input_data idata; g_framebuffer = LCD_GetFrameBuffer(&g_xres, &g_yres, &g_bpp); draw_init(); draw_end(); /* 创建队列 */ g_xQueuePlatform = xQueueCreate(10, sizeof(struct input_data)); // g_xQueueKey = xQueueCreate(10, sizeof(struct key_data)); /* 创建一个按键任务,用于获取数据 */ // xTaskCreate(KeyTask, "KeyTask", 128, NULL, osPriorityNormal, NULL); uptMove = UPT_MOVE_NONE; ball.x = g_xres / 2; ball.y = g_yres - 10; ball.velX = -0.5; ball.velY = -0.6; // ball.velX = -1; // ball.velY = -1.1; blocks = pvPortMalloc(BLOCK_COUNT); memset(blocks, 0, BLOCK_COUNT); lives = lives_origin = 3; score = 0; platformX = (g_xres / 2) - (PLATFORM_WIDTH / 2); // 创建一个挡球板任务 // xTaskCreate(platform_task, "platform_task", 128, NULL, osPriorityNormal, NULL); game1_draw(); LCD_Flush(); while (1) { /* 读取红外遥控器 */ // if (0 == IRReceiver_Read(&dev, &data)) xQueueReceive(g_xQueuePlatform, &idata, portMAX_DELAY); data = idata.val; { if (data == 0x00) { data = last_data; } if (data == 0xe0) /* Left */ { btnLeft(); } if (data == 0x90) /* Right */ { btnRight(); } last_data = data; } game1_draw(); draw_end(); vTaskDelay(50); } }
使用如上的代码在我硬件中进行测试,当我按下遥控器的左键时,挡球板跟着向左移动,当我按下遥控器的右键时,挡球板跟着向右进行移动,完成改造成使用队列读写demo。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。