STM8,延时函数

照搬原子哥的代码:http://www.openedv.com/posts/list/17347/htm

原子哥的教程有比较详细的注释和网友回复,这里仅作记录

这是基于IAR的,原子哥的帖子下面有网友回复的STVD的

main.c

#include "stm8l15x_conf.h"

volatile u8 fac_us=0;

void delay_init(u8 clk)
{
  if(clk>16)fac_us=(16-4)/4;
  else if(clk>4)fac_us=(clk-4)/4;
  else fac_us=1;
}

void delay_us(u16 nus)
{
  __asm(
"PUSH A          \n"    //1T,压栈
"DELAY_XUS:      \n"
"LD A,fac_us     \n"    //1T,fac_us加载到累加器A
"DELAY_US_1:     \n"
"NOP             \n"    //1T,nop延时
"DEC A           \n"    //1T,A--
"JRNE DELAY_US_1 \n"    //不等于0,则跳转(2T)到DELAY_US_1继续执行,若等于0,则不跳转(1T).
"NOP             \n"    //1T,nop延时
"DECW X          \n"    //1T,x--
"JRNE DELAY_XUS  \n"    //不等于0,则跳转(2T)到DELAY_XUS继续执行,若等于0,则不跳转(1T).
"POP A           \n"    //1T,出栈
);
}

void delay_ms(u32 nms)
{
  u8 t;
  if(nms>65)
  {
    t=nms/65;
    while(t--)delay_us(65000);
    nms=nms%65;
  }
  delay_us(nms*1000);
}

void Sysclk_Init(void)

{

 //HSI内部时钟,0分频,16MHZ

  CLK_HSICmd(ENABLE);
  CLK_SYSCLKSourceConfig(CLK_SYSCLKSource_HSI);   
  CLK_SYSCLKDivConfig(CLK_SYSCLKDiv_1);

}

void main(void)
{
  Sysclk_Init();
  LED_Init();
  //Uart1_Init();
  delay_init(16);  //参数16为内部时钟频率
 
  while (1)
  {
    GPIO_ToggleBits(LED1_Port, LED1_Pin);
    delay_ms(15);
  }
}

 

注1:之前用定时器2延时,每1s串口打印信息一次,用 调试助手看的时间戳会差10ms左右,以为 是我写的定时器配置的有问题,用了上面的延时之后发现是一样的,不知道是不是执行一句printf();需要10ms。

注2:上面的延时不知道怎么用示波器测量,所以就测量了下 反转LED的频率,发现延时15ms的情况下,测量出来的一个周期是29.7ms,延时应该算是准确的吧。

 

posted @ 2017-09-21 16:45  fly123  阅读(6706)  评论(1编辑  收藏  举报