microblaze之axi-timer定时器中断调试

序言:定时器是cpu不可缺少的部分,microblaze不能例外,所以需要亲自跑一遍。本文主要分享我遇到的问题和调试的经历,以及作为程序员不严谨所导致的惨痛教训。

1,axi-timer可以计数也可以定时,详细特性可以参考xilinx的手册,这里放一个它的内部框图和寄存器列表,如下:

2,在vivado里面添加axi-timer模块,连接好信号线,我首先把中断信号连接到中断控制器,然后使用一个axi-gpio模块控制他的freeze引脚,一个定时器0的generate_out控制另一个计数器的capture_in,这样可以互相验证计数和定时功能,同时把PWM通过一个IO输出,如下:

3,在SDK平台里面编写初始化代码,然后就可以在调试窗口看到打印信息,中断服务程序定时被调用。

4,以上是非常顺利的结果而已,对于第一次接触的菜鸟来说,遇到了不少问题,我真正的调试历程是首先看定时器是否正常工作了。通过调试发现定时器确实工作了,但是中断函数确没有被调用,于是想看一下interrupt信号引脚是否在翻转。

5,我把中断信号接到一个指示灯上面,我发现当定时器溢出的时候指示灯的状态改变了一次,往后再次溢出的时候就不再翻转了。于是我想到可能没有清理中断标志位,所以我手动执行中断处理函数,函数内部会清理中断状态位,结果是灯的状态正常了,说明定时器的中断信号产生了。

6,但是为何中断服务函数没有自动被执行呢,于是我把中断控制器的所有寄存器全部打印一遍,调试很久才发现进入了SW模式,没有进入HW中断模式,于是外部的中断信号肯定不会被中断控制器响应。

7,虽然这样改正了,当时中断复位函数仍然没有被执行,原因是缺少了异常注册,就是中断向量表没有初始化,如下的代码。经过修正之后,定时器中断终于稳定的工作起来了。可喜可贺。

总结:前期没有弄明白中断函数的原理,现在知道了触发的几个条件。

1,要有明确的中断信号触发interrupt输入;

2,要在中断向量表里面定义中断服务函数的地址;

3,要正确的设置中断回调函数,保证可以层层回调,最后以一个示意图总结定时器中断函数的执行原理,如下:

 

  • 7
    点赞
  • 33
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
在C语言中,你可以使用以下方式来编写DW_axi_dmac的中断服务函数: ```c #include <stdio.h> #include <stdlib.h> #include <stdint.h> #include <stdbool.h> // 假设DW_axi_dmac的寄存器地址为0x8000 #define DW_AXI_DMAC_BASE_ADDR 0x8000 // 假设DW_axi_dmac的中断寄存器地址为0x8004 #define DW_AXI_DMAC_IRQ_STATUS_REG_ADDR (DW_AXI_DMAC_BASE_ADDR + 0x4) // 假设DW_axi_dmac的中断使能寄存器地址为0x8008 #define DW_AXI_DMAC_IRQ_ENABLE_REG_ADDR (DW_AXI_DMAC_BASE_ADDR + 0x8) // 定义中断处理函数 void dw_axi_dmac_interrupt_handler(void) { // 读取中断状态寄存器 uint32_t irq_status = *(volatile uint32_t*)DW_AXI_DMAC_IRQ_STATUS_REG_ADDR; // 判断具体的中断状态 if (irq_status & 0x1) { // 中断1处理逻辑 printf("Interrupt 1 occurred.\n"); } if (irq_status & 0x2) { // 中断2处理逻辑 printf("Interrupt 2 occurred.\n"); } // 清除中断状态 *(volatile uint32_t*)DW_AXI_DMAC_IRQ_STATUS_REG_ADDR = irq_status; } int main() { // 初始化DW_axi_dmac // 使能DW_axi_dmac的中断 *(volatile uint32_t*)DW_AXI_DMAC_IRQ_ENABLE_REG_ADDR = 0x3; // 模拟中断发生 *(volatile uint32_t*)DW_AXI_DMAC_IRQ_STATUS_REG_ADDR = 0x1; // 调用中断处理函数 dw_axi_dmac_interrupt_handler(); return 0; } ``` 在这个示例中,我们假设DW_axi_dmac的寄存器地址为0x8000,并定义了中断状态寄存器地址和中断使能寄存器地址。在`dw_axi_dmac_interrupt_handler`函数中,我们首先读取中断状态寄存器的值,然后根据具体的位来判断中断状态,并执行相应的处理逻辑。最后,我们通过写入中断状态寄存器来清除中断状态。 在`main`函数中,我们初始化DW_axi_dmac,并通过写入中断使能寄存器来使能中断。然后,我们模拟中断发生,并调用`dw_axi_dmac_interrupt_handler`函数来处理中断。 请注意,以上代码仅为示例,具体的实现可能因硬件平台和驱动程序的要求而有所不同。你需要根据自己的需求和硬件平台的文档来编写适合的中断服务函数

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值