zynq 的64位全局定时器

zynq 有一个64位全局定时器,我觉得有点实用价值,比如精确的实时计算,代码运行时间计算。怎么用呢?我就google 了一下,有篇文章推荐查看2个文件。这2个文件名分别是 sleep.c, xtime_l.c。这2个文件是Vivado 安装的时候就安装好了的样例程序,我的是在这个目录下:C:\Xilinx\SDK\2015.4\data\embeddedsw\lib\bsp\standalone_v5_3\src\cortexa9查看这2个文件,就可以清楚其用法了。

实验:

在我的helloworld 工程里添加

#include "sleep.h"
#include "xtime_l.h"

然后在main 函数里添加:

main()

{    

u64 tbegin,tend;  

 long int tdiff;      

print("Begin dealay 5s\n");      

XTime_GetTime(&tbegin);      

sleep(5);    

XTime_GetTime(&tend);    

tdiff=tend-tbegin;    

print("End of dealay 5s");  

 printf("\n5s=%ldT\n1s=%d\n",tdiff,COUNTS_PER_SECOND);

}

编译连接并执行就有如下显示:

Begin dealay 5s

End of dealay 5s

5s=1666666833T

1s=333333343
程序分析:程序就是取定时器值,到tbegin, 延迟5秒, 取定时器值到tend。 2项相减得差值并显示。为了对照也显示1s 的标准值常量COUNT_PER_SECOND。程序运行的结果基本一致。定时器的频率是系统时钟的1/2, 这个是ug585 p239里写明了的。long int 原来还是32bits的, 延迟到50s 就超界了,显示负数。XTime: 其实就是 u64。我们再看看sleep.c 里 sleep 函数的定义:
s32 sleep(u32 seconds)

{  

XTime tEnd, tCur;  

XTime_GetTime(&tCur);  

tEnd  = tCur + (((XTime) seconds) * COUNTS_PER_SECOND);  

do  {    

XTime_GetTime(&tCur);

 } while (tCur < tEnd);  return 0;}

这个延迟程序直接由全局定时器控制,所以是很准确的。xtime_l.c 文件里有void XTime_SetTime(XTime Xtime_Global) void XTime_GetTime(XTime *Xtime_Global)这2个函数的原型定义,这里直接操作全局定时器的寄存器。本文程序就是根据这2个文件的内容编写出来的。有了这些内容,就可以精确地任意定时了。原来sleep(s)是精确延迟的,但单位是秒数。
--------------------- 
作者:leon_zeng0 
来源:CSDN 
原文:https://blog.csdn.net/leon_zeng0/article/details/79776175 
版权声明:本文为博主原创文章,转载请附上博文链接!

Zynq-7000系列是Xilinx的嵌入式系统SoC(System on Chip),它集成了ARM处理器和FPGA。其中的全局定时器(Global Timer)通常用于实现精确的时间测量和事件触发。下面是一个简单的例子,展示了如何在Zynq PS(Processing System)部分使用全局定时器。 ```c #include "xil_types.h" #include "xil_io.h" #include "xglutimer.h" // Xilinx Global Timer Library // 定义全局定时器配置结构体 typedef struct { u32 BaseAddress; // 全局定时器的地址 XGlutimer_Config *ConfigPtr; // 配置结构体指针 } GTimerCfg; // 初始化全局定时器函数 void InitializeGTimer(GTimerCfg *gtimer_cfg) { XGlutimer_GenInitialize(gtimer_cfg->BaseAddress, gtimer_cfg->ConfigPtr); // 初始化定时器 } // 设置定时器计数模式并启动计时 void StartTimer(GTimerCfg *gtimer_cfg, u32 period) { XGlutimer_WriteReg(gtimer_cfg->BaseAddress, XGLUTIMER_COUNT_MODE_REG_OFFSET, GLU_COUNT_UP); XGlutimer_SetTime(gtimer_cfg->BaseAddress, period); // 设置计时周期 XGlutimer_Start(gtimer_cfg->BaseAddress); // 启动计时 } // 主程序定时器回调 void TimerCallback(u32 timer_address, void *CallBackRef) { printf("Timer event occurred after %u us\n", XGlutimer_ReadReg(timer_address, XGLUTIMER_CURRENT_TIME_REG_OFFSET)); } int main(void) { GTimerCfg gtimer; // ... (填充BaseAddress和ConfigPtr) InitializeGTimer(&amp;gtimer); // 注册回调函数 XGlutimer_RegisterHandler(&amp;gtimer.BaseAddress, GLU_EVT_PERIODIC, TimerCallback, NULL); // 设置定时器周期为1秒,并启动 StartTimer(&amp;gtimer, 1000000); // 进行主循环,等待定时器事件触发 while (1) { xilSleep(1000); // 睡眠1毫秒 } return 0; } ``` 这个例子首先初始化全局定时器,然后设置其为递增计数模式,并设定一个周期。当计时器到达预设周期时,会触发注册的回调函数`TimerCallback`,打印出经过的时间。注意,实际使用时你需要根据Zynq的具体连接和配置调整`BaseAddress`和`ConfigPtr`。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值