ARM 循环阻塞延迟函数

串行驱动的关键是双方能够按照既定的时序进行检测、设置相关引脚上的电平,比如单总线、I2c这样基本的可以用GPIO模拟的时序协议,需要主从双方,必须在链路接口内严格按照微妙级的延迟单位进行时序同步。 所以,在这种对时间要求很敏感的情况下,一般都会考虑到用循环等待检测的方式来拖住时间窗,而且近似微妙的阻塞等待一般也不会对整个系统的用户体验造成太大的影响。 下面是在实际经验中积累的,微妙延迟的方法。

  1. ARM-CORTX-M0

M0 HC32F110L上的延迟,可以精准到微妙。可以用systic,也可用while阻塞循环等待。

/******************************************************************************
 * systic 
 * xus:
 *     use this delay function when time >= 20 
*******************************************************************************/
void jbf_systic_us_ex(uint32_t xus)
{ 
 	SysTick->LOAD = (xus - 18) * (JBF_SISTICK_VALUE / 1000000);
	SysTick->VAL = 0;
	SysTick->CTRL = 0x01;
	while(!(SysTick->CTRL & SysTick_CTRL_COUNTFLAG_Msk));
	SysTick->CTRL &= ~(0x01 << SysTick_CTRL_ENABLE_Pos);
}
/******************************************************************************
 * 24M 1US delay,
 * ??????????????1us??????
*******************************************************************************/
 
void jbf_delay_24M_us(void)
{
	int xus = 1;
	while( xus--)
	{
		__NOP();__NOP();
	}
	 
}

void jbf_delay_24M_xus(uint16_t xus)
{
	uint16_t j;
	while (xus--)
	{
		j = 2; //
		while (j--)
		;
	}
}
  1. ARM-CORTEX-M4

    /**********************************************************

    • DEC: 200M MPLL us delay function.
    • PARAM: n = nus
    • RETURN: none
      **********************************************************/
      void delay_nus(uint16_t n)
      {
      uint16_t j;
      while(n--)
      {
      j = DELAY_ADJUSTMENTS;//
      while(j--);
      }
      }
      extern uint32_t SystemCoreClock; /!< System clock frequency (Core clock) /
      /
    • base interval is 60 when LOAD = 1, LOAD + 1 lead the timme plus 30us
    • if the value lager than 60, and times of 30, then use this function
    • (time - 60) / 30 - 1;
    • time = 180 , load = (180 - 60) / 3 + 1 = 5;

    */

    void delay_30Tnus(uint16_t n)
    {
    uint32 _load = (n - 60) / 10 / 3 + 1;
    SysTick->LOAD = _load;
    SysTick->VAL = 0;
    SysTick->CTRL = 0x01;
    while(!(SysTick->CTRL & SysTick_CTRL_COUNTFLAG_Msk));
    SysTick->CTRL &= ~(0x01 << SysTick_CTRL_ENABLE_Pos);
    }

    /**********************************************************

    • DEC: 200M MPLL ms delay function.
    • PARAM: n = mus
    • RETURN: none
      ***********************************************************/
      void delay_nms(uint16_t n)
      {
      while(n--)
      {
      delay_nus(1000);
      }
      }

实际就两种方式比较通用的,一个是systic,这个真的要看systic分到的频率了, 如果本身频率很低,那么在微妙的精度行还是有些误差的。

另一种方式就是利用CPU的加法周期来做延迟,这个目前看来也有一定的可用性,关键是要找好指令周期的倍数。拿示波器实际调试可以作证得出最精准的延迟。

/*DELAY HC32F60,pclk 200MHZ*/

#define DELAY_ADJUSTMENTS 12

void delay_nus(uint16_t n)

{

uint16_t j;

while(n--)

{

j = DELAY_ADJUSTMENTS;//

while(j--);

}

}

相关推荐
honey ball2 小时前
LLC与反激电路设计【学习笔记】
单片机·嵌入式硬件·学习
Graceful_scenery8 小时前
STM32F103外部中断配置
stm32·单片机·嵌入式硬件
猿来不是梦11 小时前
RT_Thread内核源码分析(三)——线程
嵌入式硬件·系统架构·rt_thread操作系统
白书宇11 小时前
19.QT程序简单的运行脚本
linux·arm开发·嵌入式硬件·物联网·arm
Arciab12 小时前
51单片机入门:独立按键(02)
单片机
大梦百万秋13 小时前
嵌入式系统与单片机工作原理详解
单片机·嵌入式硬件
陌夏微秋13 小时前
硬件基础22 反馈放大电路
单片机·嵌入式硬件·硬件架构·硬件工程·智能硬件
&春风有信14 小时前
FreeRTOS之链表源码分析
c语言·数据结构·嵌入式硬件·链表
美式小田14 小时前
Altium Designer学习笔记 21.PCB板框的评估及叠层设置
笔记·嵌入式硬件·学习·ad
小郭爱学习@15 小时前
RLC串联谐振,品质因数的影响
嵌入式硬件·硬件架构·硬件工程