0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

【试用报告】沁恒CH32V307评估板体验:定时器使用基础

电子发烧友论坛 来源:未知 2023-03-28 13:05 次阅读

CH32V307系统提供了多个类型的定时器,具体可查看手册了解:



这次的分享,我们使用到的定时器为系统时基定时器和基本定时器TIM6。


闪烁使用的LED,为LED1,使用连接线将LED1和PA0连接即可;实际运行时,间隔1秒闪烁一次。



有的同学可能会说,使用Delay_Ms或Delay_Us做演示,也能实现LED闪烁呀!


但在Delay的时候,你的程序,需要在这个地方,等待Delay时间后,才会继续运行。


而使用中断,我们的程序,能够继续运行做其他的事情,等到定时中断到来的时候,才处理LED的闪烁。这样程序的处理效率将会更高。


一、系统时基定时器SysTick


现在,我们了解一下系统时基定时器SysTick。


系统时基定时器:这是内核控制器自带的一个 64 位可选递增或递减的计数器,用于产生 SYSTICK 异常(异常号:15),可专用于实时操作系统,为系统提供“心跳”节律,也可当成一个标准的 64 位计数器。具有自动重加 载功能及可编程时钟源。


通过查看提供的实例和资料,了解到如下的关键信息


1、系统的运行频率可以工作在72MHz,也可以工作在144MHz,通过system_ch32v30x.h/system_ch32v30x.c来修改:

// #define SYSCLK_FREQ_72MHz 72000000
#define SYSCLK_FREQ_144MHz 144000000

(左右移动查看全部内容)


在我的实例中,设置工作在144MHz。


2、在设置SysTick的时候,通过SysTick->CMP来设置定时中断周期:SysTick->CMP = SystemCoreClock / 1000 * 1000; //后面的1000代表1000HZ(那就是1ms进一次中断),*1000 表示 1000ms进入一次


3、有人会遇到,中断只进入一次,经过请教沁恒的陶工,了解到如下处理方法:

// 中断只进入一次的问题解决方法:
// 1. 如果使用沁恒提供的工具链,则使用如下的声明:
// void SysTick_Handler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
// 2. 如果使用通用的risc-v工具链,如我在macOS下使用赛昉提供的工具链,则使用如下的声明:
void SysTick_Handler(void) __attribute__((interrupt()));

(左右移动查看全部内容)


最终,具体的代码如下:

/*
使用VTF IRQ中断控制LED闪烁
*/


#include "debug.h"
#include "board.h"


// 中断只进入一次的问题解决方法:
// 1. 如果使用沁恒提供的工具链,则使用如下的声明:
// void SysTick_Handler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
// 2. 如果使用通用的risc-v工具链,如我在macOS下使用赛昉提供的工具链,则使用如下的声明:
void SysTick_Handler(void) __attribute__((interrupt()));


// LED状态
volatile uint16_t LED_Status = 0; // 中断里使用的变量加 volatile 可当成全局变量


// 初始化 GPIO
void GPIO_INIT(void)
{
GPIO_InitTypeDef GPIO_InitStructure = {0};


RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
}


// 初始化 SysTick 定时器
void SysTick_init(void)
{
/*配置中断优先级*/
NVIC_InitTypeDef NVIC_InitStructure = {0};
NVIC_InitStructure.NVIC_IRQChannel = SysTicK_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;//抢占式优先级
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;//响应式优先级
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//使能
NVIC_Init(&NVIC_InitStructure);


/*配置定时器*/
SysTick->CTLR= 0;
SysTick->SR = 0;
SysTick->CNT = 0;
SysTick->CMP = SystemCoreClock / 1000 * 1000; //后面的1000代表1000HZ(那就是1ms进一次中断),*1000 表示 1000ms进入一次
SysTick->CTLR= 0xf;
}


int main(void)
{
/* Initialize board components. */
BOARD_SystemClock_Config();
BOARD_IOMUX_Init();
BOARD_Peripheral_Init();


GPIO_INIT(); // 初始化 GPIO
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置中断控制器的优先级分组为 占优先级 2位 ,优先级2位。


USART_Printf_Init(115200);
printf("SystemClk:%drn", SystemCoreClock);
printf("Interrupt SysTick Testrn");


SysTick_init();
while (1)
{
}
}


// 中断服务处理
void SysTick_Handler(void)
{
SysTick->SR = 0;
LED_Status = !LED_Status; // 将 LED 状态值取反
GPIO_WriteBit(GPIOA, GPIO_Pin_0, LED_Status); // 配置 PA0 (即 LED1) 状态


printf("Toggle LED by SysTick: %drn", LED_Status);
}

(左右移动查看全部内容)


在上述代码中,关键部分如下:

  • GPIO_INIT:初始化GPIO,使用PA0;记得预先连接LED1和PA0

  • SysTick_init:SysTick初始化,主要设置中断周期

  • SysTick_Handler:中断服务处理,其内部的操作需要狠准快

除了板载的LED1(前提是连好了线)会间隔1秒闪烁一次,通过串口工具,我们也可以看到输出的调试信息:



二、基本定时器TIM6


然后,我们再来了解一下基本定时器。


基本定时器:基本定时器是一个 16 位自动装载计数器,支持 16 位可编程预分频器。可以位数模转换(DAC)提供时钟,触发 DAC 的同步电路。基本定时器之间是互相独立的,互不共享任何资源。


CH32V307提供了两个基本定时器TIM6和TIM7,用法是一样的,下面的实例中,使用TIM6。


通过官方提供的实例和资料了解到,基本定时器使用的过程中,有两个设置是最重要的:

  • TIM_Prescaler:定时器预分频器设置,时钟源经该预分频器才是定时器时钟。

  • TIM_Period:定时器周期,实际就是设定自动重载寄存器的值。


这两个参数结合就能实现实际所需要的定时。


根据定时器时钟的频率,比如时钟的频率是144MHz,可以理解为1秒钟MCU会自己数144M次,预分频系数就是将频率分割,比如分频系数是144,则该时钟的频率会变成144MHZ/144=1MHz,但是在设置的时候要注意,数值应该是144-1。


为了让 LED1间隔 1 秒闪烁一次,我们需要让定时器 1 秒溢出,要计数 144M * 1 = 144M 个时钟周期,而定时器只有16位,最大65535,所以这是不够的。因此,需要用到预分频器,设分频系数为 14400,可以得到 10KHz 的定时器时钟,这样设置计数值 10000 就可以做到 1s 定时。


最终具体的代码如下:

/*
使用基本定时器TIM6控制LED闪烁
*/


#include "debug.h" // 包含 CH32V307 的头文件,C 标准单元库和delay()函数
#include "board.h"


// 中断只进入一次的问题解决方法:
// 1. 如果使用沁恒提供的工具链,则使用如下的声明:
// void TIM6_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));
// 2. 如果使用通用的risc-v工具链,如我在macOS下使用赛昉提供的工具链,则使用如下的声明:
void TIM6_IRQHandler(void) __attribute__((interrupt()));


// LED状态
volatile uint16_t LED_Status = 0; // 中断里使用的变量加 volatile 可当成全局变量


// 初始化 GPIO
void GPIO_INIT(void)
{
GPIO_InitTypeDef GPIO_InitStructure = {0};


RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
}


// 初始化定时器 TIM6
void TIM6_Init(u16 arr, u16 psc)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure;


RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM6, ENABLE);


TIM_TimeBaseInitStructure.TIM_Period = arr;
TIM_TimeBaseInitStructure.TIM_Prescaler = psc;
TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;
TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Down;
TIM_TimeBaseInit(TIM6, &TIM_TimeBaseInitStructure);


TIM_ITConfig(TIM6, TIM_IT_Update, ENABLE);
TIM_ARRPreloadConfig(TIM6, ENABLE);
TIM_Cmd(TIM6, ENABLE);
}


// 初始化定时器中断
void Interrupt_Init(void)
{
NVIC_InitTypeDef NVIC_InitStructure = {0};
NVIC_InitStructure.NVIC_IRQChannel = TIM6_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; //抢占优先级
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2; //子优先级
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
}


int main(void)
{
/* Initialize board components. */
BOARD_SystemClock_Config();
BOARD_IOMUX_Init();
BOARD_Peripheral_Init();


GPIO_INIT(); // 初始化 GPIO
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //设置中断控制器的优先级分组为 占优先级 2位 ,优先级2位。


USART_Printf_Init(115200);


TIM6_Init(10000 - 1, 14400 - 1); // 初始化定时器,让 LED 1 秒闪烁一次,我们需要让定时器 1 秒溢出,要计数 `144M * 1 = 144M` 个时钟周期,而定时器只有16位,这是不够的。需要用到预分频器,设分频系数为 14400,可以得到 10KHz 的定时器时钟,这样设置计数值 10000 就可以做到 1s 定时。
Interrupt_Init(); //初始化定时器中断


printf("SystemClk:%drn", SystemCoreClock);
printf("Interrupt TIM6 Testrn");


while (1)
{
}
}


// 中断服务处理
void TIM6_IRQHandler(void)
{
TIM_ClearFlag(TIM6, TIM_FLAG_Update); //清除标志位
LED_Status = !LED_Status; // 将 LED 状态值取反
GPIO_WriteBit(GPIOA, GPIO_Pin_0, LED_Status); // 配置 PE11 (即 LED1) 状态


printf("Toggle LED by TIM6: %drn", LED_Status);
}

(左右移动查看全部内容)


在上述代码中,关键部分如下:

  • GPIO_INIT:初始化GPIO,使用PA0;记得预先连接LED1和PA0

  • TIM6_init:TIM6初始化,主要设置分频和时间周期

  • Interrupt_Init:终端初始化

  • TIM6_IRQHandler中断服务处理,其内部的操作同样需要狠准快


除了板载的LED1(前提是连好了线)会间隔1秒闪烁一次,通过串口工具,我们也可以看到输出的调试信息:






声明本文由电子发烧友社区发布,转载请注明以上来源。如需社区合作及入群交流,请添加微信EEFans0806,或者发邮箱liuyong@huaqiu.com。


更多热点文章阅读

  • 龙芯架构首款面向嵌入式应用的开发板,2K500开发应用实例

  • ARM架构国产MCU移植!国民技术N32系列开源移植样例合集

  • RK3568!四核64位ARMv8.2A架构,汇聚编译源码及实战样例

  • OpenHarmony开源鸿蒙大赛作品集:基于RK2206开发板20+成熟应用案例!

  • 24Bit ADC高精度低功耗MCU 医疗级别设备参考设计(附上高精确算法+电路原理图)


原文标题:【试用报告】沁恒CH32V307评估板体验:定时器使用基础

文章出处:【微信公众号:电子发烧友论坛】欢迎添加关注!文章转载请注明出处。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 电子技术
    +关注

    关注

    18

    文章

    889

    浏览量

    55892
  • 电子发烧友论坛

    关注

    4

    文章

    197

    浏览量

    1062
  • ch32
    +关注

    关注

    0

    文章

    73

    浏览量

    633

原文标题:【试用报告】沁恒CH32V307评估板体验:定时器使用基础

文章出处:【微信号:gh_9b9470648b3c,微信公众号:电子发烧友论坛】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    使用CH32V307驱动ADS1256输出数据不稳定怎么解决?

    使用ch32v307驱动ads1256输出不正常,目前的问题是启动之后一次读50个数据,通道0和1输入接了下拉电阻,无输入时读到的数据有时是-8191194左右,有时是6左右,很不稳定,然后还会
    发表于 11-13 06:39

    BQ24040应用报告

    电子发烧友网站提供《BQ24040应用报告.pdf》资料免费下载
    发表于 10-14 09:28 0次下载
    BQ24040应<b class='flag-5'>用报告</b>

    TAS5504-5142V4EVM应用报告

    电子发烧友网站提供《TAS5504-5142V4EVM应用报告.pdf》资料免费下载
    发表于 09-30 11:45 0次下载
    TAS5504-5142<b class='flag-5'>V</b>4EVM应<b class='flag-5'>用报告</b>

    龙芯2K0300蜂鸟试用报告

    龙芯2K0300蜂鸟试用报告 一、试用环境 操作系统和框架:Linux+QT5.15 交叉编译工具链
    发表于 09-13 18:00

    高速数据转换应用报告

    电子发烧友网站提供《高速数据转换应用报告.pdf》资料免费下载
    发表于 09-09 14:57 0次下载
    高速数据转换应<b class='flag-5'>用报告</b>

    【龙芯2K0300蜂鸟试用】龙芯2K0300蜂鸟试用报告

    龙芯2K0300蜂鸟试用报告 1.拆箱、上电开机 2.自带CH340串口通讯芯片,不需要再用串口工具,已经板载了 3.主要由JW5079A、SW34201B、FORESEE
    发表于 08-09 11:58

    ch32v307_RTT】1、使用RT-Thread studio新建工程

    【前言】 ch32v307有成熟的RT-Thread 支持,使用RT-Thread studio可以快速的建立工程,我将分享系列文章,这一篇是如何快速建立工程。 【前题】 安装好RT-Thread
    发表于 06-29 18:25

    WCH32V307体验tcpserver

    的端口,然后发送一段内容,可以接收到发送的内容: 【总结】 这个块ch32V307拿到好久了,这次重新体验了一下tcpserver的示例,下次再体验mqtt示例。
    发表于 04-30 17:24

    的蓝牙分析仪怎么用?

    我在的的店里看到有一个蓝牙分析仪,不知道他是如何用的,是不是所有的蓝牙都可以分析,还是只针对他们的协议栈有用。
    发表于 04-29 06:58

    CH32V307V评估】之HarmonyOS运行

    我们知道CH32V307是基于32位RISC-V内核设计的大容量通用微控制器,搭载V4F内核,支持单精度浮点指令集,具有更高的运算性能。支持内置PHY收发器的USB2.0高速接口(480Mbps
    发表于 04-28 23:38

    使用MounRiver Studio快速创建CH32V208开发环境

    CH32V208WBU6 4、生成如下工程 ,在工程中已经给出了测试的程序: 5、点击下载按键下的config菜单可以弹出配置,配置如下: 6、然后下载到开发,打开串口助手,就可以收到打印的信息了: 【总结】
    发表于 04-18 11:54

    国产RISC-V MCU推荐

    这颗芯片,众望所归 众多网友都推荐了微(WCH)的CH32V307。29447945表示最近正好在用RISC-V架构的芯片(CH32V307
    发表于 04-17 11:00

    分享CH32X035评估说明及参考应用例程

    定时器等丰富外设资源,可轻松构建电机方案;同时广泛应用于光伏、储能电源等相关领域。 CH32X035评估说明及参考应用例程: /elecplay.php?action=show&am
    发表于 03-18 11:04

    CH32X035可以运行RTT操作系统吗?

    CH32X035支持USB通讯+PD电源双功能,集成PIOC、I2C、多组运放/比较器/定时器等丰富片上资源,助您推开Type-C的大门,开启PDUSB特色应用创意之旅。 不知道他
    发表于 03-18 10:59

    CH32V303_305_307 数据手册

    电子发烧友网站提供《CH32V303_305_307 数据手册.rar》资料免费下载
    发表于 01-25 09:08 1次下载