为什么xQueueSendToBack调用被阻止?

时间:2019-03-18 09:06:20

标签: c embedded freertos

我正在STM32F103C8T6(在Blue-Pill板上)上学习FreeRTOS。 我正在尝试使用队列和任务。

#include <FreeRTOS.h>
#include <libopencm3/stm32/gpio.h>
#include <libopencm3/stm32/rcc.h>
#include <queue.h>
#include <task.h>

static QueueHandle_t queue;

static void
task_receive(void *args __attribute__((unused)))
{
  bool nothing;
  while (1)
  {
    if (xQueueReceive(queue, &nothing, 10) == pdPASS)
      gpio_set(GPIOC, GPIO13); // Turn off
    else
      taskYIELD(); // Yeld so that other taks can run
  }
}

static void
task_send(void *args __attribute__((unused)))
{
  bool nothing = false;
  while (1)
  {
    gpio_clear(GPIOC, GPIO13); // Turn on
    vTaskDelay(pdMS_TO_TICKS(100));
    xQueueSendToBack(queue, &nothing, portMAX_DELAY);
    vTaskDelay(pdMS_TO_TICKS(1000));
  }
}

int
main(void)
{
  rcc_clock_setup_in_hse_8mhz_out_72mhz();

  // Blue-Pill led
  rcc_periph_clock_enable(RCC_GPIOC);
  gpio_set_mode(
    GPIOC,
    GPIO_MODE_OUTPUT_2_MHZ,
    GPIO_CNF_OUTPUT_PUSHPULL,
    GPIO13);
  gpio_set(GPIOC, GPIO13); // Turn off (polarity of the led is inversed!)

  queue = xQueueCreate(32, sizeof(bool));
  if (queue == 0)
  {
    while (1)
    {
      gpio_toggle(GPIOC, GPIO13);
      for (uint32_t i = 0; i < 80000; ++i)
        __asm__("nop");
    };
  }
  xTaskCreate(task_receive, "RECEIVE", 200, NULL, configMAX_PRIORITIES-1, NULL);
  xTaskCreate(task_send, "SEND", 200, NULL, configMAX_PRIORITIES-2, NULL);
  vTaskStartScheduler();
  while(1);
  return 0;
}

预期行为:

  • main

    • 配置时钟
    • 为蓝色药丸LED配置GPIO
    • 关闭LED
    • 创建队列
    • 检查是否正确创建了队列:如果未正确闪烁,则LED会永远闪烁。
    • 安排两个任务
    • 运行调度程序
  • task_send(无限循环)

    • 打开LED
    • 等待100毫秒
    • 将消息推送到队列中(内容在这里无关紧要)
    • 等待1秒
  • task_receive(无限循环)

    • 检查消息是否在队列中
      • 是:关闭LED
      • 否:收益

我希望先打开LED指示灯100毫秒,然后关闭900毫秒。

真实行为:led一直亮着,程序的执行似乎在xQueueSendToBack处受阻。

为什么通话阻塞?


FreeRTOSConfig.h

#define configUSE_PREEMPTION        1
#define configUSE_IDLE_HOOK     0
#define configUSE_TICK_HOOK     0
#define configCPU_CLOCK_HZ      ( ( unsigned long ) 72000000 )  
#define configSYSTICK_CLOCK_HZ      ( configCPU_CLOCK_HZ / 8 ) /* fix for vTaskDelay() */
#define configTICK_RATE_HZ      ( ( TickType_t ) 1000 )
#define configMAX_PRIORITIES        ( 5 )
#define configMINIMAL_STACK_SIZE    ( ( unsigned short ) 128 )
#define configTOTAL_HEAP_SIZE       ( ( size_t ) ( 17 * 1024 ) )
#define configMAX_TASK_NAME_LEN     ( 16 )
#define configUSE_TRACE_FACILITY    0
#define configUSE_16_BIT_TICKS      0
#define configIDLE_SHOULD_YIELD     1
#define configUSE_MUTEXES       1

/* Co-routine definitions. */
#define configUSE_CO_ROUTINES       0
#define configMAX_CO_ROUTINE_PRIORITIES ( 2 )

/* Set the following definitions to 1 to include the API function, or zero
to exclude the API function. */

#define INCLUDE_vTaskPrioritySet        1
#define INCLUDE_uxTaskPriorityGet       1
#define INCLUDE_vTaskDelete             1
#define INCLUDE_vTaskCleanUpResources   0
#define INCLUDE_vTaskSuspend            1
#define INCLUDE_vTaskDelayUntil         1
#define INCLUDE_vTaskDelay              1

/* This is the raw value as per the Cortex-M3 NVIC.  Values can be 255
(lowest) to 0 (1?) (highest). */
#define configKERNEL_INTERRUPT_PRIORITY         255
/* !!!! configMAX_SYSCALL_INTERRUPT_PRIORITY must not be set to zero !!!!
See http://www.FreeRTOS.org/RTOS-Cortex-M3-M4.html. */
#define configMAX_SYSCALL_INTERRUPT_PRIORITY    191 /* equivalent to 0xb0, or priority 11. */


/* This is the value being used as per the ST library which permits 16
priority values, 0 to 15.  This must correspond to the
configKERNEL_INTERRUPT_PRIORITY setting.  Here 15 corresponds to the lowest
NVIC value of 255. */
#define configLIBRARY_KERNEL_INTERRUPT_PRIORITY 15

2 个答案:

答案 0 :(得分:2)

您的task_receive优先级高于task_send优先级。如果没有更高优先级的任务,taskYIELD将一次又一次地运行相同的调用任务。

要实现所需的功能,请尝试按以下方式更改task_receive

static void
task_receive(void *args __attribute__((unused)))
{
  bool nothing;
  while (1)
  {
    if (xQueueReceive(queue, &nothing, portMAX_DELAY) == pdPASS)
      gpio_set(GPIOC, GPIO13); // Turn off
  }
}

有关taskYIELD的更多信息,请参阅以下内容。

https://www.freertos.org/a00020.html#taskYIELD

答案 1 :(得分:0)

通过将编译器更新到最新版本解决了该问题。

Kubuntu 18.04随附arm-none-eabi-gcc (15:6.3.1+svn253039-1build1) 6.3.1 20170620,使用此编译器,代码无法正常工作。

memcpy似乎是代码中有问题的函数调用,它是FreeRTOS在将元素添加到队列时调用的。

如果我使用Version 8-2018-q4-major Linux 64-bit编译器,则代码可以正常执行。可以在这里下载:https://developer.arm.com/open-source/gnu-toolchain/gnu-rm/downloads