搜索
bottom↓
回复: 10

新人学习STM32,明明设置的300ms延时,烧进去后成了2,3秒

[复制链接]

出0入0汤圆

发表于 2014-8-20 21:38:12 | 显示全部楼层 |阅读模式
用的库函数编写的。。参照正点原子的开发指南写的,按照上面一模一样写的,可是烧进去的延时莫名增加了近十倍

#include "delay.h"
#include "sys.h"
//////////////////////////////////////////////////////////////////////////////////          
//如果使用ucos,则包括下面的头文件即可.
#if SYSTEM_SUPPORT_UCOS
#include "includes.h"                                        //ucos 使用          
#endif
//////////////////////////////////////////////////////////////////////////////////         
//本程序只供学习使用,未经作者许可,不得用于其它任何用途
//ALIENTEK STM32开发板
//使用SysTick的普通计数模式对延迟进行管理
//包括delay_us,delay_ms
//正点原子@ALIENTEK
//技术论坛:www.openedv.com
//修改日期:2012/9/2
//版本:V1.5
//版权所有,盗版必究。
//Copyright(C) 广州市星翼电子科技有限公司 2009-2019
//All rights reserved
//********************************************************************************
//V1.2修改说明
//修正了中断中调用出现死循环的错误
//防止延时不准确,采用do while结构!

//V1.3修改说明
//增加了对UCOSII延时的支持.
//如果使用ucosII,delay_init会自动设置SYSTICK的值,使之与ucos的TICKS_PER_SEC对应.
//delay_ms和delay_us也进行了针对ucos的改造.
//delay_us可以在ucos下使用,而且准确度很高,更重要的是没有占用额外的定时器.
//delay_ms在ucos下,可以当成OSTimeDly来用,在未启动ucos时,它采用delay_us实现,从而准确延时
//可以用来初始化外设,在启动了ucos之后delay_ms根据延时的长短,选择OSTimeDly实现或者delay_us实现.

//V1.4修改说明 20110929
//修改了使用ucos,但是ucos未启动的时候,delay_ms中中断无法响应的bug.
//V1.5修改说明 20120902
//在delay_us加入ucos上锁,防止由于ucos打断delay_us的执行,可能导致的延时不准。
//////////////////////////////////////////////////////////////////////////////////          
static u8  fac_us=0;//us延时倍乘数
static u16 fac_ms=0;//ms延时倍乘数
#ifdef OS_CRITICAL_METHOD         //如果OS_CRITICAL_METHOD定义了,说明使用ucosII了.
//systick中断服务函数,使用ucos时用到
void SysTick_Handler(void)
{                                  
        OSIntEnter();                //进入中断
    OSTimeTick();       //调用ucos的时钟服务程序               
    OSIntExit();        //触发任务切换软中断
}
#endif

//初始化延迟函数
//当使用ucos的时候,此函数会初始化ucos的时钟节拍
//SYSTICK的时钟固定为HCLK时钟的1/8
//SYSCLK:系统时钟
void delay_init()         
{

#ifdef OS_CRITICAL_METHOD         //如果OS_CRITICAL_METHOD定义了,说明使用ucosII了.
        u32 reload;
#endif
        SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK_Div8);        //选择外部时钟  HCLK/8
        fac_us=SystemCoreClock/8000000;        //为系统时钟的1/8  
         
#ifdef OS_CRITICAL_METHOD         //如果OS_CRITICAL_METHOD定义了,说明使用ucosII了.
        reload=SystemCoreClock/8000000;                //每秒钟的计数次数 单位为K          
        reload*=1000000/OS_TICKS_PER_SEC;//根据OS_TICKS_PER_SEC设定溢出时间
                                                        //reload为24位寄存器,最大值:16777216,在72M下,约合1.86s左右       
        fac_ms=1000/OS_TICKS_PER_SEC;//代表ucos可以延时的最少单位          
        SysTick->CTRL|=SysTick_CTRL_TICKINT_Msk;           //开启SYSTICK中断
        SysTick->LOAD=reload;         //每1/OS_TICKS_PER_SEC秒中断一次       
        SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk;           //开启SYSTICK   
#else
        fac_ms=(u16)fac_us*1000;//非ucos下,代表每个ms需要的systick时钟数   
#endif
}                                                                    

#ifdef OS_CRITICAL_METHOD        //使用了ucos
//延时nus
//nus为要延时的us数.                                                                                      
void delay_us(u32 nus)
{               
        u32 ticks;
        u32 told,tnow,tcnt=0;
        u32 reload=SysTick->LOAD;        //LOAD的值                     
        ticks=nus*fac_us;                         //需要的节拍数                           
        tcnt=0;
        told=SysTick->VAL;                //刚进入时的计数器值
        while(1)
        {
                tnow=SysTick->VAL;       
                if(tnow!=told)
                {            
                        if(tnow<told)tcnt+=told-tnow;//这里注意一下SYSTICK是一个递减的计数器就可以了.
                        else tcnt+=reload-tnow+told;            
                        told=tnow;
                        if(tcnt>=ticks)break;//时间超过/等于要延迟的时间,则退出.
                }  
        };                                                                             
}
//延时nms
//nms:要延时的ms数
void delay_ms(u16 nms)
{       
        if(OSRunning==TRUE)//如果os已经在跑了            
        {                  
                if(nms>=fac_ms)//延时的时间大于ucos的最少时间周期
                {
                           OSTimeDly(nms/fac_ms);//ucos延时
                }
                nms%=fac_ms;                                //ucos已经无法提供这么小的延时了,采用普通方式延时   
        }
        delay_us((u32)(nms*1000));        //普通方式延时,此时ucos无法启动调度.
}
#else//不用ucos时
//延时nus
//nus为要延时的us数.                                                                                      
void delay_us(u32 nus)
{               
        u32 temp;                     
        SysTick->LOAD=nus*fac_us; //时间加载                           
        SysTick->VAL=0x00;        //清空计数器
        SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk ;          //开始倒数         
        do
        {
                temp=SysTick->CTRL;
        }
        while(temp&0x01&&!(temp&(1<<16)));//等待时间到达   
        SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk;       //关闭计数器
        SysTick->VAL =0X00;       //清空计数器         
}
//延时nms
//注意nms的范围
//SysTick->LOAD为24位寄存器,所以,最大延时为:
//nms<=0xffffff*8*1000/SYSCLK
//SYSCLK单位为Hz,nms单位为ms
//对72M条件下,nms<=1864
void delay_ms(u16 nms)
{                                     
        u32 temp;                  
        SysTick->LOAD=(u32)nms*fac_ms;//时间加载(SysTick->LOAD为24bit)
        SysTick->VAL =0x00;           //清空计数器
        SysTick->CTRL|=SysTick_CTRL_ENABLE_Msk ;          //开始倒数  
        do
        {
                temp=SysTick->CTRL;
        }
        while(temp&0x01&&!(temp&(1<<16)));//等待时间到达   
        SysTick->CTRL&=~SysTick_CTRL_ENABLE_Msk;       //关闭计数器
        SysTick->VAL =0X00;       //清空计数器                      
}
#endif

上面是delay.c


下面是主程序
#include "LED.h"
#include "delay.h"
#include "sys.h"

int main(void)
{
        delay_init();
        LED_Init();
        while(1)
        {
                DS0=0;
                DS1=1;
                delay_ms(300);
                DS0=1;
                DS1=0;
                delay_ms(300);
        }
}



阿莫论坛20周年了!感谢大家的支持与爱护!!

你熬了10碗粥,别人一桶水倒进去,淘走90碗,剩下10碗给你,你看似没亏,其实你那10碗已经没有之前的裹腹了,人家的一桶水换90碗,继续卖。说白了,通货膨胀就是,你的钱是挣来的,他的钱是印来的,掺和在一起,你的钱就贬值了。

出0入0汤圆

 楼主| 发表于 2014-8-20 21:43:26 | 显示全部楼层
重新复制了一遍库函数又好了,实在搞不懂了。。

出0入0汤圆

发表于 2014-8-22 16:48:25 | 显示全部楼层
时钟设置检查下。

出0入90汤圆

发表于 2014-8-22 17:18:47 | 显示全部楼层
估计是外部HSE没起震。结果使用的是内部HSI,这种情况下,默认的是没有倍频的。实际时钟速度就变成了8MHZ,而不是72MHZ,所以时间相差了接近10倍。

出0入0汤圆

发表于 2014-8-23 01:50:33 来自手机 | 显示全部楼层
honami520 发表于 2014-8-22 17:18
估计是外部HSE没起震。结果使用的是内部HSI,这种情况下,默认的是没有倍频的。实际时钟速度就变成了8MHZ, ...

这个解释的合理
头像被屏蔽

出0入0汤圆

发表于 2014-8-23 09:33:19 | 显示全部楼层
提示: 作者被禁止或删除 内容自动屏蔽

出0入0汤圆

 楼主| 发表于 2014-8-23 09:38:52 | 显示全部楼层
honami520 发表于 2014-8-22 17:18
估计是外部HSE没起震。结果使用的是内部HSI,这种情况下,默认的是没有倍频的。实际时钟速度就变成了8MHZ, ...

没起振是硬件原因还是软件设置原因?我只是用了官方的库文件,也没对库文件修改过。反正重新复制一遍库文件,又好了。

出0入90汤圆

发表于 2014-8-23 11:00:49 | 显示全部楼层
Edwardwei 发表于 2014-8-23 09:38
没起振是硬件原因还是软件设置原因?我只是用了官方的库文件,也没对库文件修改过。反正重新复制一遍库文 ...

库里面有个初始化的函数。那个函数里面在进入main之前,就已经执行了的。时钟都是在这个里面进行配置的。
一般STM32F103默认都是HSE为8M,然后9倍频PLL。但是他会判断HSE是否起震,如果没有,就使用内部HSI,但是就没有PLL了。最后出来的就是8M内部晶振。
你这个现象,有时候可以,有时候不行。肯定是硬件问题。
运气好的话,再不会出现了,运气不好,偶尔给你来一次。
建议换个8M晶振,同时把它的2个20pf电容也给换一次。

出0入0汤圆

发表于 2015-8-4 20:58:31 | 显示全部楼层
honami520 发表于 2014-8-22 17:18
估计是外部HSE没起震。结果使用的是内部HSI,这种情况下,默认的是没有倍频的。实际时钟速度就变成了8MHZ, ...

大哥,就是屌啊,一眼看出来了。其实这是正点原子的启动文件把主函数启动之前的SystemInit给注释掉

出0入0汤圆

发表于 2015-8-4 21:00:58 | 显示全部楼层
这是正点原子的启动文件把主函数启动之前的SystemInit给注释掉。你只需要把注释删除就好

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有帐号?注册

x

出0入0汤圆

发表于 2015-8-4 21:53:51 来自手机 | 显示全部楼层
连初始化代码都要写个寄存器版本的,呵呵呵,这个还蛮搞笑的
回帖提示: 反政府言论将被立即封锁ID 在按“提交”前,请自问一下:我这样表达会给举报吗,会给自己惹麻烦吗? 另外:尽量不要使用Mark、顶等没有意义的回复。不得大量使用大字体和彩色字。【本论坛不允许直接上传手机拍摄图片,浪费大家下载带宽和论坛服务器空间,请压缩后(图片小于1兆)才上传。压缩方法可以在微信里面发给自己(不要勾选“原图),然后下载,就能得到压缩后的图片。注意:要连续压缩2次才能满足要求!!】。另外,手机版只能上传图片,要上传附件需要切换到电脑版(不需要使用电脑,手机上切换到电脑版就行,页面底部)。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

手机版|Archiver|amobbs.com 阿莫电子技术论坛 ( 粤ICP备2022115958号, 版权所有:东莞阿莫电子贸易商行 创办于2004年 (公安交互式论坛备案:44190002001997 ) )

GMT+8, 2024-9-29 09:18

© Since 2004 www.amobbs.com, 原www.ourdev.cn, 原www.ouravr.com

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