使用下面的stopwatch_delay(ticks)来完成延迟.它使用STM32的DWT_CYCCNT寄存器,该寄存器专门用于计算位于地址0xE0001004的实际时钟周期.

要验证延迟准确性(请参阅main),您可以调用STOPWATCH_START,运行stopwatch_delay(ticks),然后调用STOPWATCH_STOP并使用CalcNanosecondsFromStopwatch(m_nStart,m_nStop)进行验证.根据需要调整刻度.

uint32_t m_nStart; //DEBUG Stopwatch start cycle counter value

uint32_t m_nStop; //DEBUG Stopwatch stop cycle counter value

#define DEMCR_TRCENA 0x01000000

/* Core Debug registers */

#define DEMCR (*((volatile uint32_t *)0xE000EDFC))

#define DWT_CTRL (*(volatile uint32_t *)0xe0001000)

#define CYCCNTENA (1<<0)

#define DWT_CYCCNT ((volatile uint32_t *)0xE0001004)

#define CPU_CYCLES *DWT_CYCCNT

#define STOPWATCH_START { m_nStart = *((volatile unsigned int *)0xE0001004);}

#define STOPWATCH_STOP { m_nStop = *((volatile unsigned int *)0xE0001004);}

static inline void stopwatch_reset(void)

{

/* Enable DWT */

DEMCR |= DEMCR_TRCENA;

*DWT_CYCCNT = 0;

/* Enable CPU cycle counter */

DWT_CTRL |= CYCCNTENA;

}

static inline uint32_t stopwatch_getticks()

{

return CPU_CYCLES;

}

static inline void stopwatch_delay(uint32_t ticks)

{

uint32_t end_ticks = ticks + stopwatch_getticks();

while(1)

{

if (stopwatch_getticks() >= end_ticks)

break;

}

}

uint32_t CalcNanosecondsFromStopwatch(uint32_t nStart, uint32_t nStop)

{

uint32_t nDiffTicks;

uint32_t nClkTicksPerMicrosec;

nDiffTicks = nStop - nStart;

nDiffTicks *= 1000; // Scale diff by 1000.

nClkTicksPerMicrosec = SystemCoreClock / 1000000; // Convert (clkTicks/sec) to (clkTicks/microsec), SystemCoreClock = 168000000

return nDiffTicks / nClkTicksPerMicrosec; // nanosec = (ticks * 1000) / (clkTicks/microsec)

}

void main(void)

{

int timeDiff = 0;

stopwatch_reset();

STOPWATCH_START;

run_my_function();

STOPWATCH_STOP;

timeDiff = CalcNanosecondsFromStopwatch(m_nStart, m_nStop);

printf("My function took %d nanoseconds\n", timeDiff);

}

Logo

魔乐社区(Modelers.cn) 是一个中立、公益的人工智能社区,提供人工智能工具、模型、数据的托管、展示与应用协同服务,为人工智能开发及爱好者搭建开放的学习交流平台。社区通过理事会方式运作,由全产业链共同建设、共同运营、共同享有,推动国产AI生态繁荣发展。

更多推荐