硬汉嵌入式论坛

 找回密码
 立即注册
查看: 3405|回复: 16
收起左侧

[FreeRTOS] freertos xTimerCreate时间不准

[复制链接]

2

主题

11

回帖

17

积分

新手上路

积分
17
发表于 2018-1-18 11:20:35 | 显示全部楼层 |阅读模式
本帖最后由 wtyiyou 于 2018-1-18 11:22 编辑

我的处理器是STM32F103.主频64M,内部RC倍频做的。现在发现一个问题,用freertos  xTimerCreate创建一个周期软件定时器,定时周期是1000.也就是周期1S.定时器回调函数里面有一个串口信息输出,发现在1S里面有8条串口信息输出,每次差不多都是8条。正确的应该是1S差不多就是1条,在误差的情况下2条也可以。这是什么原因呀,希望大家帮忙给解决一下呀
回复

使用道具 举报

28

主题

73

回帖

162

积分

初级会员

积分
162
发表于 2018-1-18 11:29:36 | 显示全部楼层
你FreeRTOSconfig里面配置的系统时钟是多少啊
回复

使用道具 举报

1万

主题

6万

回帖

10万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
107049
QQ
发表于 2018-1-18 11:30:26 | 显示全部楼层
确认两个问题
1. 你的主频时钟确认是在64MHz时钟下工作,确定方式是在RCC库函数里面有一个时钟获取函数,获取下看看。
2. 确认你配置的系统时钟节拍是多少,是不是配置的1000Hz,或者其他。在FreeRTOSConfig.h文件里面设置的。
回复

使用道具 举报

2

主题

11

回帖

17

积分

新手上路

积分
17
 楼主| 发表于 2018-1-18 11:38:30 | 显示全部楼层
本帖最后由 wtyiyou 于 2018-1-18 11:41 编辑

我确认了一下:  在这个文件里面配置的是64M,节拍配置的是1000
为了确认是不是64M,在运行RTOS前我加了一个串口信息输出,输出就是64000000下面图片是配置

FreeRTOSConfig.h

FreeRTOSConfig.h


回复

使用道具 举报

1万

主题

6万

回帖

10万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
107049
QQ
发表于 2018-1-18 11:43:02 | 显示全部楼层
wtyiyou 发表于 2018-1-18 11:38
我确认了一下:FreeRTOSConfig.h  在这个文件里面配置的是64M,节拍配置的是1000
为了确认是不是64M,在运行 ...

可以这样验证,你在你建立的任务里面使用函数vTaskDelay延迟1s看看打印几次,如果也是8次左右,可以肯定是你的主频时钟或者你的FreeRTOSConfig.h配置有问题,如果正常就是你的定时器创建有问题。
回复

使用道具 举报

2

主题

11

回帖

17

积分

新手上路

积分
17
 楼主| 发表于 2018-1-18 11:48:09 | 显示全部楼层
回复

使用道具 举报

2

主题

11

回帖

17

积分

新手上路

积分
17
 楼主| 发表于 2018-1-18 11:56:03 | 显示全部楼层
    AutoReloadTimer_Handle=xTimerCreate((const char*                    )"Timer",
                                                         (TickType_t                    )1000,
                                                      (UBaseType_t                    )pdTRUE,
                                                     (void*                                )Data_Save_Timer_ID,
                                                   (TimerCallbackFunction_t)AutoReloadCallback);                                
                                                                                                                                                               
        xTimerStart(AutoReloadTimer_Handle,200);               
回复

使用道具 举报

2

主题

11

回帖

17

积分

新手上路

积分
17
 楼主| 发表于 2018-1-18 11:56:20 | 显示全部楼层
这是定时器创建和启动
回复

使用道具 举报

2

主题

11

回帖

17

积分

新手上路

积分
17
 楼主| 发表于 2018-1-18 14:24:24 | 显示全部楼层
本帖最后由 wtyiyou 于 2018-1-18 14:45 编辑

@eric2013  你说的对,vTaskDelay延时也不对,主频变为8M, vTaskDelay(2000)输出一个调试信息,结果2S是8条,主频变为64M,vTaskDelay(1000)结果1S也是8条,你帮忙看看FreeRTOSConfig.h哪有问题
/***************************************************************************************************************/
/*                                        FreeRTOS基础配置配置选项                                              */
/***************************************************************************************************************/
#define configUSE_PREEMPTION                                        1                                               //1使用抢占式内核,0使用协程
#define configUSE_TIME_SLICING                                        1                                                                //1使能时间片调度(默认式使能的)
#define configUSE_PORT_OPTIMISED_TASK_SELECTION        1                                       //1启用特殊方法来选择下一个要运行的任务
                                                                                                                                                                //一般是硬件计算前导零指令,如果所使用的
                                                                                                                                                                //MCU没有这些硬件指令的话此宏应该设置为0!
#define configUSE_TICKLESS_IDLE                                        0                                       //1启用低功耗tickless模式
#define configUSE_QUEUE_SETS                                        1                                       //为1时启用队列
#define configCPU_CLOCK_HZ                                                ((unsigned long)SystemCoreClock)//CPU频率  SystemCoreClock是64M我测试了
#define configTICK_RATE_HZ                                                ((TickType_t)1000 )             //时钟节拍频率,这里设置为1000,周期就是1ms
#define configMAX_PRIORITIES                                        (16)                                    //可使用的最大优先级
#define configMINIMAL_STACK_SIZE                                ((unsigned short)128)                   //空闲任务使用的堆栈大小
#define configMAX_TASK_NAME_LEN                                        (30)                                    //任务名字字符串长度

#define configUSE_16_BIT_TICKS                                        0                                       //系统节拍计数器变量数据类型,
                                                                                                                                                                //1表示为16位无符号整形,0表示为32位无符号整形
#define configIDLE_SHOULD_YIELD                                        1                                       //为1时空闲任务放弃CPU使用权给其他同优先级的用户任务
#define configUSE_TASK_NOTIFICATIONS            0                                       //为1时开启任务通知功能,默认开启
#define configUSE_MUTEXES                                                0                                       //为1时使用互斥信号量
#define configQUEUE_REGISTRY_SIZE                                8                                       //不为0时表示启用队列记录,具体的值是可以
                                                                                                                                                                //记录的队列和信号量最大数目。
#define configCHECK_FOR_STACK_OVERFLOW                        0                                       //大于0时启用堆栈溢出检测功能,如果使用此功能
                                                                                                                                                                //用户必须提供一个栈溢出钩子函数,如果使用的话
                                                                                                                                                                //此值可以为1或者2,因为有两种栈溢出检测方法。
#define configUSE_RECURSIVE_MUTEXES                                0                                       //为1时使用递归互斥信号量
#define configUSE_MALLOC_FAILED_HOOK                        0                                       //1使用内存申请失败钩子函数
#define configUSE_APPLICATION_TASK_TAG                        0                       
#define configUSE_COUNTING_SEMAPHORES                        0                                       //为1时使用计数信号量

/***************************************************************************************************************/
/*                                FreeRTOS与内存申请有关配置选项                                                */
/***************************************************************************************************************/
#define configSUPPORT_DYNAMIC_ALLOCATION        1                                       //支持动态内存申请
#define configTOTAL_HEAP_SIZE                                        ((size_t)(30*1024))                     //系统所有总的堆大小

/***************************************************************************************************************/
/*                                FreeRTOS与钩子函数有关的配置选项                                              */
/***************************************************************************************************************/
#define configUSE_IDLE_HOOK                                                0                                       //1,使用空闲钩子;0,不使用
#define configUSE_TICK_HOOK                                                0                                       //1,使用时间片钩子;0,不使用


#define configUSE_TRACE_FACILITY                                0                                       //为1启用可视化跟踪调试
#define configUSE_STATS_FORMATTING_FUNCTIONS        0                                       //与宏configUSE_TRACE_FACILITY同时为1时会编译下面3个函数
                                                                                                                                                                //prvWriteNameToBuffer(),vTaskList(),
                                                                                                                                                           //vTaskGetRunTimeStats()
/***************************************************************************************************************/
/*                                FreeRTOS与协程有关的配置选项                                                  */
/***************************************************************************************************************/
#define configUSE_CO_ROUTINES                                 0                                       //为1时启用协程,启用协程以后必须添加文件croutine.c
#define configMAX_CO_ROUTINE_PRIORITIES         ( 2 )                                   //协程的有效优先级数目

/***************************************************************************************************************/
/*                                FreeRTOS与软件定时器有关的配置选项                                            */
/***************************************************************************************************************/
#define configUSE_TIMERS                                        1                               //为1时启用软件定时器
#define configTIMER_TASK_PRIORITY                        (configMAX_PRIORITIES-1)        //软件定时器优先级
#define configTIMER_QUEUE_LENGTH                        5                               //软件定时器队列长度
#define configTIMER_TASK_STACK_DEPTH                (configMINIMAL_STACK_SIZE*2)    //软件定时器任务堆栈大小

/***************************************************************************************************************/
/*                                FreeRTOS可选函数配置选项                                                      */
/***************************************************************************************************************/
#define INCLUDE_xTaskGetSchedulerState          1                       
#define INCLUDE_vTaskPrioritySet                        1
#define INCLUDE_uxTaskPriorityGet                        1
#define INCLUDE_vTaskDelete                                        1
#define INCLUDE_vTaskCleanUpResources                1
#define INCLUDE_vTaskSuspend                                1
#define INCLUDE_vTaskDelayUntil                                1
#define INCLUDE_vTaskDelay                                        1
#define INCLUDE_eTaskGetState                                1
#define INCLUDE_xTimerPendFunctionCall                1
#define        INCLUDE_xTaskGetHandle                                        1
#define INCLUDE_uxTaskGetStackHighWaterMark                1




/***************************************************************************************************************/
/*                                FreeRTOS与中断有关的配置选项                                                  */
/***************************************************************************************************************/
#ifdef __NVIC_PRIO_BITS
        #define configPRIO_BITS                       __NVIC_PRIO_BITS
#else
        #define configPRIO_BITS                       4                  
#endif

#define configLIBRARY_LOWEST_INTERRUPT_PRIORITY                        15                      //中断最低优先级
#define configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY        5                       //系统可管理的最高中断优先级
#define configKERNEL_INTERRUPT_PRIORITY                 ( configLIBRARY_LOWEST_INTERRUPT_PRIORITY << (8 - configPRIO_BITS) )
#define configMAX_SYSCALL_INTERRUPT_PRIORITY         ( configLIBRARY_MAX_SYSCALL_INTERRUPT_PRIORITY << (8 - configPRIO_BITS) )

/***************************************************************************************************************/
/*                                FreeRTOS与中断服务函数有关的配置选项                                          */
/***************************************************************************************************************/
#define vPortSVCHandler         SVC_Handler
#define xPortPendSVHandler         PendSV_Handler
#define xPortSysTickHandler SysTick_Handler


这是FreeRTOSConfig.h  这里面的配置内容.
回复

使用道具 举报

1万

主题

6万

回帖

10万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
107049
QQ
发表于 2018-1-18 14:58:45 | 显示全部楼层
先将你FreeRTOSConfig.h文件里面的SystemCoreClock修改成实际数值,如果是64MHz,就添64000000。

另外系统时钟的获取要调用如下函数来获取。
  1. RCC_ClocksTypeDef  rcc_clocks;

  2. RCC_GetClocksFreq(&rcc_clocks);
复制代码

RCC_ClocksTypeDef  的定义如下:
  1. /* Exported types ------------------------------------------------------------*/
  2. typedef struct
  3. {
  4.   uint32_t SYSCLK_Frequency; /*!<  SYSCLK clock frequency expressed in Hz */
  5.   uint32_t HCLK_Frequency;   /*!<  HCLK clock frequency expressed in Hz   */
  6.   uint32_t PCLK1_Frequency;  /*!<  PCLK1 clock frequency expressed in Hz  */
  7.   uint32_t PCLK2_Frequency;  /*!<  PCLK2 clock frequency expressed in Hz  */
  8. }RCC_ClocksTypeDef;
复制代码



回复

使用道具 举报

2

主题

11

回帖

17

积分

新手上路

积分
17
 楼主| 发表于 2018-1-18 15:23:30 | 显示全部楼层
本帖最后由 wtyiyou 于 2018-1-18 16:05 编辑

微信图片_20180118152222.png
获取看了就是我设置的64000000上面是串口输出,上面的runing   是vTaskDelay(1000);输出的  管理员,你帮忙看看啥原因,系统可以跑起来,就是时间不对







回复

使用道具 举报

1万

主题

6万

回帖

10万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
107049
QQ
发表于 2018-1-18 16:50:18 | 显示全部楼层
wtyiyou 发表于 2018-1-18 15:23
获取看了就是我设置的64000000上面是串口输出,上面的runing   是vTaskDelay(1000);输出的  管理员,你帮 ...

务必用我上面说的函数RCC_GetClocksFreq(&rcc_clocks);获取。
回复

使用道具 举报

2

主题

11

回帖

17

积分

新手上路

积分
17
 楼主| 发表于 2018-1-18 16:51:28 | 显示全部楼层
就是用你那个函数获取的,然后串口打印的
回复

使用道具 举报

1万

主题

6万

回帖

10万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
107049
QQ
发表于 2018-1-18 16:59:30 | 显示全部楼层
wtyiyou 发表于 2018-1-18 16:51
就是用你那个函数获取的,然后串口打印的

最后一招,我只剩下这一招了。你找个示波器或者逻辑分析仪,然后在port.C文件的滴答定时器中断函数xPortSysTickHandler里面加个IO翻转,看看是不是1000Hz。
回复

使用道具 举报

2

主题

11

回帖

17

积分

新手上路

积分
17
 楼主| 发表于 2018-1-18 17:07:14 | 显示全部楼层
那出现这个现象还有别的原因吗?
回复

使用道具 举报

2

主题

11

回帖

17

积分

新手上路

积分
17
 楼主| 发表于 2018-1-18 21:35:19 | 显示全部楼层
本帖最后由 wtyiyou 于 2018-1-18 22:27 编辑

void xPortSysTickHandler( void )
{
        /* The SysTick runs at the lowest interrupt priority, so when this interrupt
        executes all interrupts must be unmasked.  There is therefore no need to
        save and then restore the interrupt mask value as its value is already
        known - therefore the slightly faster vPortRaiseBASEPRI() function is used
        in place of portSET_INTERRUPT_MASK_FROM_ISR(). */
        vPortRaiseBASEPRI();
        {
                /* Increment the RTOS tick. */
                if( xTaskIncrementTick() != pdFALSE )
                {
                        /* A context switch is required.  Context switching is performed in
                        the PendSV interrupt.  Pend the PendSV interrupt. */
                        portNVIC_INT_CTRL_REG = portNVIC_PENDSVSET_BIT;
                }
        }
        vPortClearBASEPRIFromISR();
}
@eric2013  你好,我在port.c 这个文件里面添加了一个头文件所有的都正常了,任何一个头文件都可以,指示灯是正常,其他的输出也正常了,我定时器定时1s,串口1S就输出1次了,这是什么原因呀?

回复

使用道具 举报

1万

主题

6万

回帖

10万

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
107049
QQ
发表于 2018-1-19 01:56:03 | 显示全部楼层
wtyiyou 发表于 2018-1-18 21:35
void xPortSysTickHandler( void )
{
        /* The SysTick runs at the lowest interrupt priority, s ...

一般这种比较神奇的情况,你的程序模板应该有地方有问题,你查找下,或者重新移植下。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|小黑屋|Archiver|手机版|硬汉嵌入式论坛

GMT+8, 2024-5-16 02:11 , Processed in 0.216173 second(s), 28 queries .

Powered by Discuz! X3.4 Licensed

Copyright © 2001-2023, Tencent Cloud.

快速回复 返回顶部 返回列表