高云FPGA系列教程(6):ARM定时器使用

### 文章目录

  • [@[toc]](#文章目录 @[toc] 1. ARM定时器简介 2. FPGA配置 3. 常用函数 4. MCU程序设计 5. 工程下载)
  • [1. ARM定时器简介](#文章目录 @[toc] 1. ARM定时器简介 2. FPGA配置 3. 常用函数 4. MCU程序设计 5. 工程下载)
  • [2. FPGA配置](#文章目录 @[toc] 1. ARM定时器简介 2. FPGA配置 3. 常用函数 4. MCU程序设计 5. 工程下载)
  • [3. 常用函数](#文章目录 @[toc] 1. ARM定时器简介 2. FPGA配置 3. 常用函数 4. MCU程序设计 5. 工程下载)
  • [4. MCU程序设计](#文章目录 @[toc] 1. ARM定时器简介 2. FPGA配置 3. 常用函数 4. MCU程序设计 5. 工程下载)
  • [5. 工程下载](#文章目录 @[toc] 1. ARM定时器简介 2. FPGA配置 3. 常用函数 4. MCU程序设计 5. 工程下载)

本文是高云FPGA系列教程第6篇文章

本篇文章介绍片上ARM Cortex-M3硬核处理器定时器外设的使用,演示定时器溢出中断的配置方法,基于TangNano 4K开发板。

参考文档:Gowin_EMPU(GW1NS-4C)软件编程 参考手册

1. ARM定时器简介

GW1NSR-4C ARM处理器部分共有两个定时器,这两个定时器都是32位的,可以产生中断请求,也可以对外部输入信号进行计数,这两个定时器都挂载在APB1总线上,时钟源是PCLK1。

2. FPGA配置

和串口不同,定时器不用手动开启,直接在MCU程序中使用即可。

3. 常用函数

以下是定时器驱动库常用的一些函数,都比较容易理解,使用也很简单:

c 复制代码
//定时器外设初始化,指定溢出值
void TIMER_Init(TIMER_TypeDef* TIMERx,TIMER_InitTypeDef* TIMER_InitStruct)
//启动定时器
void TIMER_StartTimer(TIMER_TypeDef* TIMERx)
//停止定时器
void TIMER_StopTimer(TIMER_TypeDef* TIMERx)
//读取中断状态
ITStatus TIMER_GetIRQStatus(TIMER_TypeDef* TIMERx)
//清除中断状态
void TIMER_ClearIRQ(TIMER_TypeDef* TIMERx)
//读取重载值(溢出值)
uint32_t TIMER_GetReload(TIMER_TypeDef* TIMERx)
//设定重载值
void TIMER_SetReload(TIMER_TypeDef* TIMERx,uint32_t value)
//获取当前计数值
uint32_t TIMER_GetValue(TIMER_TypeDef* TIMERx)
//手动指定计数值
void TIMER_SetValue(TIMER_TypeDef* TIMERx,uint32_t value)
//使能中断
void TIMER_EnableIRQ(TIMER_TypeDef* TIMERx)
//禁止中断
void TIMER_DisableIRQ(TIMER_TypeDef* TIMERx)
//外部输入使能
void TIMER_SelExtEnable(TIMER_TypeDef *TIMER)
//选择外部输入时钟
void TIMER_SelExtClock(TIMER_TypeDef *TIMER)

GW1NSR-4C定时器一般配置步骤如下:

c 复制代码
1. 定时器初始化。指定重载值(溢出值)
2. NVIC使能定时器中断源,并指定优先级
3. 开启定时器中断
4. 启动定时器
5. 定时器中断服务函数实现,并清除中断

4. MCU程序设计

MCU程序可以使用Keil或GWD进行设计,参考上一篇文章提供的工程模板

首先是定时器和中断控制器初始化,开启中断并指定优先级:

c 复制代码
int timer_init(void)
{
    TIMER_InitTypeDef tmr0;
    NVIC_InitTypeDef InitTypeDef_NVIC;
    
    tmr0.Reload = PCLK1/2;	//APB1 CLK, 500ms定时器
    TIMER_Init(TIMER0, &tmr0);

    //Interrupt priority configuration
    NVIC_PriorityGroupConfig(NVIC_PriorityGroup_3);

    //Enable Timer0 interrupt
    InitTypeDef_NVIC.NVIC_IRQChannel = TIMER0_IRQn;
    InitTypeDef_NVIC.NVIC_IRQChannelPreemptionPriority = 1;
    InitTypeDef_NVIC.NVIC_IRQChannelSubPriority = 1;
    InitTypeDef_NVIC.NVIC_IRQChannelCmd = ENABLE;
    NVIC_Init(&InitTypeDef_NVIC);

    TIMER_EnableIRQ(TIMER0);//enable timer0 interrupt register
    TIMER_StartTimer(TIMER0);//startup timer0

    return 0;
}

中断服务函数实现:

c 复制代码
//Timer0 interrupt handler
void TIMER0_Handler(void)
{
    static uint16_t cnt = 0;
    if(TIMER_GetIRQStatus(TIMER0) != RESET)
    {
        cnt++;
        printf("timer0 int...%d\r\n", cnt);        
        TIMER_ClearIRQ(TIMER0);
    }
}

需要把gw1ns4c_it.c文件中的定时器0中断服务函数注释掉,否则编译会报错

主函数调用:

c 复制代码
int main(void)
{
	delay_init();
	uart0_init(115200);
    timer_init();
    
	printf("SystemCoreClock = %d\r\n", SystemCoreClock);
	printf("APB1 CLK = %d\r\n", PCLK1);
	printf("APB2 CLK = %d\r\n", PCLK2);
	printf("AHB CLK  = %d\r\n", HCLK);
	printf("Hello GW1NSR-4C SoC(ARM Cortex-M3)\r\n");
    printf("TIMER0 Interrupt Example\r\n");
    
	while(1)
	{

	}
}

查询每个外设挂载在哪个总线上,可以打开gw1ns4c.h文件查看:

或者在EMPU配置界面查看:

各总线的时钟频率,可以打开system_gw1ns4c.c文件查看:

c 复制代码
//system_gw1ns4c.c
#define __SYSTEM_CLOCK    (60000000UL)	   /* 60MHz */

uint32_t SystemCoreClock = __SYSTEM_CLOCK;/*!< System Clock Frequency (Core Clock)*/
uint32_t PCLK1 = __SYSTEM_CLOCK;          /*!< APB1 Clock Frequency */
uint32_t PCLK2 = __SYSTEM_CLOCK;          /*!< APB2 Clock Frequency */
uint32_t HCLK = __SYSTEM_CLOCK;           /*!< AHB Clock Frequency */

这里和FPGA工程中配置的保持一致,这里我设置的是60MHz。

编译生成Bin文件,打开下载工具,指定FPGA的fs文件和MCU的bin文件进行下载。

打开串口调试助手,可以看到串口数据输出的时间间隔大概是500ms。

5. 工程下载

基于TangNano 4K的配套工程,包括Keil和GMD开发环境,都可以正常使用。

本文是高云FPGA系列教程第6篇文章

相关推荐
贝塔实验室1 天前
FPGA 动态重构配置流程
驱动开发·fpga开发·硬件架构·硬件工程·射频工程·fpga·基带工程
tiantianuser2 天前
RDMA简介5之RoCE v2队列
fpga开发·verilog·fpga·rdma·高速传输·rocev2
ALINX技术博客2 天前
【新品解读】一板多能,AXRF49 定义新一代 RFSoC FPGA 开发平台
射频工程·fpga·amd·rfsoc·alinx
才鲸嵌入式2 天前
01 Ubuntu20.04下编译QEMU8.2.4,交叉编译32位ARM程序,运行ARM程序的方法
linux·c语言·单片机·嵌入式·arm·qemu·虚拟机
迎风打盹儿2 天前
FPGA仿真中阻塞赋值(=)和非阻塞赋值(<=)区别
verilog·fpga·阻塞赋值·非阻塞赋值·testbench仿真
tiantianuser3 天前
RDMA简介3之四种子协议对比
verilog·fpga·vivado·rdma·高速传输
Rocky4013 天前
JavaEE->多线程:定时器
java·开发语言·多线程·定时器
Rocky4015 天前
javaEE->多线程:线程池
java·运维·服务器·线程池·多线程·定时器
XINVRY-FPGA9 天前
XCVP1902-2MSEVSVA6865 Xilinx FPGA Versal Premium SoC/ASIC
嵌入式硬件·安全·阿里云·ai·fpga开发·云计算·fpga
謓泽10 天前
【STM32】按键控制LED & 光敏传感器控制蜂鸣器
stm32·单片机·arm