当前位置:首页 > 单片机 > 单片机
[导读] 前天学了下stm的systick,发现还满好用的,可以用来精确定时.以前在用CVAVR的时候发现里面的delay.h非常好用.于是,利用stm32的SysTick做了个精确的延时头函数. SysTick的配置在void delay_init(u8 SYS

前天学了下stm的systick,发现还满好用的,可以用来精确定时.以前在用CVAVR的时候发现里面的delay.h非常好用.于是,利用stm32的SysTick做了个精确的延时头函数.


SysTick的配置在void delay_init(u8 SYSCLK);里面输入的参数SYSCLK是你配置的系统时钟,比如72M,我就调用delay_init(72);然后就完成了对SysTick的初始化设置.在后面调用delay_ms(u32 Nms);delay_us(u32 Nus);就可以得到很准确的延时.


注意:


1,delay_us(u32 Nus);在Nus值很小的时候,误差比较大,我仿真的时候delay_us(1);实际上得到的是延时了1.5us左右.理论上Nus的值越大,越准确.


2,delay_ms(u32 Nms);的参数不能太大!更具你系统时钟来确定.72M的时候Nms的最大值是1864.在其他晶振(<72M)条件下,该值会变大.计算方法:Nms<=0xffffff*8/SYSCLK.


在包括了这个头函数之后可以得到很准确的延时,也不会产生中断,其他中断可以打断delay_us,delay_ms的执行.


#ifndef __DELAY_H

#define __DELAY_H

//使用SysTick的普通计数模式对延迟进行管理

//包括delay_us,delay_ms

//正点原子@SCUT

//2008/12/13

static u8 fac_us=0;//us延时倍乘数

static u16 fac_ms=0;//ms延时倍乘数

//初始化延迟函数

void delay_init(u8 SYSCLK)

{

SysTick->CTRL&=0xfffffffb;//选择内部时钟 HCLK/8

fac_us=SYSCLK/8;

fac_ms=(u16)fac_us*1000;

}

//延时Nms

//注意Nms的范围

//Nms<=0xffffff*8/SYSCLK

//对72M条件下,Nms<=1864

void delay_ms(u16 nms)

{

SysTick->LOAD=(u32)nms*fac_ms; //时间加载

SysTick->CTRL|=0x01; //开始倒数

while(!(SysTick->CTRL&(1<<16))); //等待时间到达

SysTick->CTRL&=0XFFFFFFFE; //关闭计数器

SysTick->VAL=0X00000000; //清空计数器

}

//延时us

void delay_us(u32 Nus)

{

SysTick->LOAD=Nus*fac_us; //时间加载

SysTick->CTRL|=0x01; //开始倒数

while(!(SysTick->CTRL&(1<<16)));//等待时间到达

SysTick->CTRL=0X00000000; //关闭计数器

SysTick->VAL=0X00000000; //清空计数器

}

#endif


另外的一种方法 :


/*---------------------------------

延时模块函数

说明:只需在工程中加入delay.c和delay.h

文件,即可用 Delayms(__IO uint32_t nTime);

Delayus(__IO uint32_t nTime)

-----------------------------------*/

#ifndef __DELAY_H

#define __DELAY_H

#include "stm32f10x.h"

/*---------------------------------

描 述:参数1即为1ms,1000即为1s;只有几

us的误差;

-----------------------------------*/

extern void Delayms(__IO uint32_t nTime);


/*---------------------------------

描 述:参数1即为1us,1000即为1ms;只有几

us的误差;

-----------------------------------*/

extern void Delayus(__IO uint32_t nTime);

/*---------------------------------

函数名:延时调整形式的delaynus函数,

描 述:参数1即为1us,1000即为1ms;

-----------------------------------*/

extern void delay_nus(unsigned long n);

/*---------------------------------

函数名:延时调整形式的delaynms函数,

描 述:参数1即为1ms,1000即为1s;

-----------------------------------*/

extern void delay_nms(unsigned long n);

#endif


C 函数


/*---------------------------------

延时模块函数

说明:只需在工程中加入delay.c和delay.h

文件,即可用 Delayms(__IO uint32_t nTime);

Delayus(__IO uint32_t nTime)

-----------------------------------*/

#include"delay.h"

static __IO uint32_t TimingDelay;

/* Private function prototypes -----------------------------------------------*/

/*---------------------------------

函数名:ms延时函数

描 述:参数1即为1ms,1000即为1s;只有几

us的误差;

-----------------------------------*/

void Delayms(__IO uint32_t nTime)

{

while(SysTick_Config(SystemCoreClock/1000));

TimingDelay = nTime;

while(TimingDelay != 0);

SysTick->CTRL=0x00; //关闭计数器

SysTick->VAL =0X00; //清空计数器

}

/*---------------------------------

函数名:us延时函数

描 述:参数1即为1us,1000即为1ms;只有几

us的误差;

-----------------------------------*/

void Delayus(__IO uint32_t nTime)

{

while(SysTick_Config(SystemCoreClock/1000000));

TimingDelay = nTime;

while(TimingDelay != 0);

SysTick->CTRL=0x00; //关闭计数器

SysTick->VAL =0X00; //清空计数器

}

/*---------------------------------

函数名:延时辅助函数

描 述:

-----------------------------------*/

void TimingDelay_Decrement(void)

{

if (TimingDelay != 0x00)

{

TimingDelay--;

}

}

/*---------------------------------

函数名:systick的中断函数

描 述:参数1即为1us,1000即为1ms;只有几

us的误差;

-----------------------------------*/

void SysTick_Handler(void)

{

TimingDelay_Decrement();

}

/*---------------------------------

函数名:延时调整形式的delaynus函数,

描 述:参数1即为1us,1000即为1ms;

-----------------------------------*/

void delay_nus(unsigned long n)

{

unsigned long j;

while(n--)

{

j=12;

while(j--);

}

}

/*---------------------------------

函数名:延时调整形式的delaynms函数,

描 述:参数1即为1ms,1000即为1s;

-----------------------------------*/

void delay_nms(unsigned long n)

{

while(n--)

delay_nus(1030);

}


本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

在Linux环境下的C语言编程中,函数的可重入性(Reentrancy)是一个至关重要的概念。它直接关系到多线程或多任务环境下程序的稳定性和可靠性。所谓可重入函数,是指一个函数可以被多个线程或任务安全地调用,而不会导致数...

关键字: Linux 库函数

这篇文章想分享笔者在嵌入式开发过程中常用的库函数,他们的使用方法,使用场景,使用好处以及头文件位置。1#include一般我在写C代码的时候都会包含此头文件,因为一旦包含此文件后,你就可以畅快的使用bool数据类型,而不...

关键字: 库函数

关注、星标公众号,直达精彩内容来源:知乎作者:Clarence这篇文章想分享笔者在嵌入式开发过程中常用的库函数,他们的使用方法,使用场景,使用好处以及头文件位置。1#include一般我在写C代码的时候都会包含此头文件,...

关键字: 库函数

基本概念阐述memcpy和memmove都是C语言的库函数,相比于 strcpy和 strncpy只能针对于字符类型的数组(),这两个函数可以拷贝其他类型的数组,对于 memcpy和 memmove的区别是什么呢?这里,...

关键字: 库函数 ov

在嵌入式Linux的C语言开发中,C语言的基本编程依然是最重要的内容。

关键字: 嵌入式 C语言 库函数

自秋季以来,8英寸晶圆代工产能紧缺,报价调涨,MCU、MOS,TDDI,闪存,面板等电子元器件进入了愈演愈烈的涨价模式。ST单片机的热门型号涨幅更是一度达到了60%,并且交期长达3个月。国产替换需求趋势迫在眉睫,世强硬创...

关键字: 世强元件电商 st MCU

蝴蝶扇动翅膀,便可引发彼岸的暴风雨。STM32这只“蝴蝶”不仅掀起了MCU的风暴,还走进了工程师和千家万户。

关键字: st MCU mpu

一、什么是物联网及物联网终端   物联网   目前业界对物联网还没有一个完全统一的概念,但普遍认可的概念是通过射频识别(RFID)、红外感应器、全球定位系统(GPS)、激光扫描器、环境传感器、图

关键字: st stk6153

汽车可能是用户必须操作的最尖端的电子设备。 由于其日益复杂, 很有必要不断提高司机和乘客们的舒适和方便的特点。 这些功能涉及了如车顶和门区控制系统,暖通空调系统,雨刷和座椅控制系统,照明控制系统

关键字: 汽车电子 st 电子控制 机电一体化

IBM、意法半导体(ST)与 Shaspa 宣佈将携手推动云端和行动运算在智慧家庭领域的发展,让设备厂商和服务供应商能为消费者带来创新的居家功能管理和互动方式,透过手势识别和语音辨识等多种用户介

关键字: IBM st 智能家居 shaspa
关闭