你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

利用DWT进行精准us延迟

[复制链接]
卿枫凌 提问时间:2018-12-20 16:50 /
10ST金币
我使用STM32L151芯片,采用内部MSI时钟,4.194MHz,在利用DWT进行us延时的时候发现差距特别大,代码如下
void PUB_DelayInit(WORD32 SYSClock)
{
    SystemCoreClock = SYSClock;

    DEM_CR      |= (unsigned int)DEM_CR_TRCENA;     // 开启DWT调试
        DWT_CYCCNT   = (unsigned int)0u;                // 清零
        DWT_CR      |= (unsigned int)DWT_CR_CYCCNTENA;  // 使能
}

void PUB_DelayUs(WORD16 us)
{
    WORD32 tCnt, tDelayCnt;
        WORD32 tStart;
               
        tStart = DWT_CYCCNT;                            // 刚进入时的计数器值
        tCnt = 0;
        tDelayCnt = us * (SystemCoreClock/1000000.0);        // 需要的节拍数                      

        while(tCnt < tDelayCnt)
        {
                tCnt = DWT_CYCCNT - tStart;               
        }
}


先进行PUB_DelayInit(4194000);然后调用PUB_DelayUs(5),示波器观察是10us,不知道为什么会出现这种情况呢?和网上说的精度差距有点大

收藏 评论3 发布时间:2018-12-20 16:50

举报

3个回答
radio2radio 回答时间:2018-12-21 00:40:21
本帖最后由 radio2radio 于 2018-12-21 00:42 编辑

我的精准us延迟是这样得到的:


  1. //-----uS delay ( >=1us)---------------
  2. void vDelayUS(uint32_t u32DelayUS)
  3. {
  4.         uint32_t i;
  5.        
  6.         if (u32DelayUS <= 15)
  7.         {
  8.                 switch(u32DelayUS)
  9.                 {
  10.                         case 1:
  11.                                 __NOP();
  12.                                 __NOP();
  13.                                 __NOP();
  14.                                 return;
  15.                        
  16.                         case 2:
  17.                                 i = 3;
  18.                                 __NOP();
  19.                                 break;
  20.                         case 3:
  21.                                 i = 6;
  22.                                 __NOP();
  23.                                 __NOP();
  24.                                 __NOP();
  25.                                 break;
  26.                         case 4:
  27.                                 i = 10;
  28.                                 __NOP();
  29.                                 __NOP();
  30.                                 break;
  31.                         default:
  32.                                 i = (u32DelayUS - 2) << 2;
  33.                                 i++;
  34.                                 i++;
  35.                 }
  36. loop:        if (--i > 0) goto loop;
  37.                 return;
  38.         }
  39.         // >=16us
  40.         u32DelayUS = (DELAY_US_FACTOR * u32DelayUS) >> 12;
  41.         for (i = 0;  i < u32DelayUS; i++) {}
  42. }
复制代码

上面的代码是在NXP的MCU上面,调试以后得到的。 注意,
1. CPU的时钟频率,直接影响结果。
2. 编译器的优化选择项,直接影响结果。
3. 使用时,必须用示波器,校准。


评分

参与人数 1蝴蝶豆 +3 收起 理由
STMCU + 3

查看全部评分

卿枫凌 回答时间:2018-12-21 10:12:34
radio2radio 发表于 2018-12-21 00:40
我的精准us延迟是这样得到的:

还有这种操作,厉害了
七哥 回答时间:2018-12-21 11:07:06
本帖最后由 toofree 于 2018-12-21 11:12 编辑

楼主,设置延时时间长一点。
就这么个程序,做乘除法都得一段时间,还有程序进出的进栈、出栈时间。或者把IO电平反转放在紧挨着while的前后,看看相隔多长时间。
另外SystemCoreClock不应该是PUB_DelayInit(4194000)用户指定的吧,除非确定实际时间就是这个,一般是库函数中设置了。

评分

参与人数 1蝴蝶豆 +2 收起 理由
STMCU + 2

查看全部评分

所属标签

相似问题

关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版