CW32定时器及其中断介绍

以CW32L083为例,其定时器分为6种:

基本定时器:CW32L083 内部集成 3 个基本定时器 (BTIM),每个 BTIM 完全独立且功能完全相同,各包含一个 16bit 自动重装 载计数器并由一个可编程预分频器驱动。BTIM 支持定时器模式、计数器模式、触发启动模式和门控模式 4 种工作模式,支持溢出事件触发中断请求和 DMA 请求。得益于对触发信号的精细处理设计,使得 BTIM 可以由硬件自 动执行触发信号的滤波操作,还能令触发事件产生中断和 DMA 请求。

低功耗定时器:CW32L083 内部集成 1 个 16 位低功耗定时器(LPTIM),可以以很低的功耗实现定时或对外部脉冲计数的功能。通过选择合适的时钟源和触发信号,可以实现系统低功耗休眠时将MCU唤醒的功能。LPTIM 内部具有一个比较寄存器,可实现比较输出和 PWM 输出,并可以控制输出波形的极性。此外,LPTIM 还可以与正交编码器连接,自动 实现递增计数和递减计数。

通用定时器:CW32L083 内部集成 4 个通用定时器(GTIM),每个 GTIM 完全独立且功能完全相同,各包含一个16bit 自动重 装载计数器并由一个可编程预分频器驱动。GTIM 支持定时器模式、计数器模式、触发启动模式和门控模式 4 种基本工作模式,每组带4 路独立的捕获 / 比较通道,可以测量输入信号的脉冲宽度(输入捕获)或者产生输出波形(输出比较和 PWM)。

高级定时器:高级定时器 (ATIM) 由一个 16 位的自动重载计数器和 7 个比较单元组成,并由一个可编程的预分频器驱动。ATIM 支持 6 个独立的捕获 / 比较通道,可实现 6 路独立 PWM 输出或 3 对互补 PWM 输出或对 6 路输入进行捕获。可 用于基本的定时 / 计数、测量输入信号的脉冲宽度和周期、产生输出波形(PWM、单脉冲、插入死区时间的互补 PWM 等)。

独立看门狗定时器 (IWDT):CW32L083 内部集成独立看门狗定时器 (IWDT),使用专门的内部 RC 时钟源 RC10K,可避免运行时受到外部因素 影响。一旦启动 IWDT,用户需要在规定时间间隔内对 IWDT 的计数器进行重载,否则计数器溢出会触发复位或 产生中断信号。IWDT 启动后,可停止计数。可选择在深度休眠模式下 IWDT 保持运行或暂停计数。专门设置的键值寄存器,可以锁定 IWDT 的关键寄存器,防止寄存器被意外修改。

窗口看门狗定时器 (WWDT):CW32L083 内部集成窗口看门狗定时器 (WWDT),用户需要在设定的时间窗口内进行刷新,否则将触发系统复位。WWDT 通常被用来监测有严格时间要求的程序执行流程,防止由外部干扰或未知条件造成应用程序的执行异常, 导致发生系统故障。

01

CW32定时器中断

定时器中断是由CW32中的定时器引起的中断,所谓中断就是程序正常顺序执行的时候,出现了突发事件,CPU停止当前的程序的执行,转去处理突发事件,处理完毕后又返回原程序被中断的位置继续执行。

02

CW32定时器中断源

定时器主要中断源包括如下:

基本定时器(BTIM): 计数器(ARR/TOP)溢出中断、计数器触发中断。

低功耗定时器(LPTIM): 计数方向反向中断、ARR更新完成中断、比较寄存器更新完成中断、计数器触发中断、ARR自动重载匹配中断、比较匹配中断。

通用定时器(GTIM): 编码器计数方向变化中断、CHx捕获比较中断、计数器下溢中断、计数器触发中断、计数器ARR溢出中断。

高级定时器(ATIM): CHxA/B捕获比较中断、CHxA/B捕获数据丢失中断、计数器上溢中断、计数器下溢中断、刹车中断、CH4比较匹配中断、事件更新中断。

更新事件, 触发事件。

独立看门狗定时器(IWDT): 计数器溢出中断。

窗口看门狗定时器(WWDT):计数器溢出中断。

各中断源的含义详细描述参见对应产品的用户手册内容。

03

实际操作

以CW32L083为例,控制基本定时器BTIM1以固定的时间间隔产生中断,并在ARR溢出中断中控制口线电平翻转。

1. RCC时钟初始化

void RCC_Configuration(void)
{
RCC_HSI_Enable(RCC_HSIOSC_DIV6);
//系统时钟设置为HSI时钟6分频,8MHz, PCLK、HCLK不分频,PCLK=HCLK=SysClk=8MHz
__RCC_BTIM_CLK_ENABLE();
__RCC_GPIOB_CLK_ENABLE();
}

2.初始化GPIO口

void GPIO_Configuration(void)
{
GPIO_InitTypeDef GPIO_InitStruct = {0};
GPIO_InitStruct.IT = GPIO_IT_NONE;
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
GPIO_InitStruct.Pins = GPIO_PIN_8;
GPIO_Init(CW_GPIOB, &GPIO_InitStruct);
}

3.配置嵌套矢量中断控制器

void NVIC_Configuration(void)
{
    __disable_irq();
    NVIC_EnableIRQ(BTIM1_IRQn);
    __enable_irq();
} 

4.主函数

int32_t main(void)
{
BTIM_TimeBaseInitTypeDef BTIM_TimeBaseInitStruct = {0};

/* System Clocks Configuration */
    RCC_Configuration();

    /* NVIC Configuration */
    NVIC_Configuration();

    /* GPIO Configuration */
    GPIO_Configuration();

BTIM_TimeBaseInitStruct.BTIM_Mode = BTIM_Mode_TIMER;
/* 工作模式,00:定时器模式01:计数器模式10:触发计数模式11:门控计数模式  */
BTIM_TimeBaseInitStruct.BTIM_Period = 49999;
/* 计数重载周期,取值范围0x0000 到 0xFFFF.  */
    BTIM_TimeBaseInitStruct.BTIM_Prescaler = 7;    // 8分频
/*8分频, 预分配系数,取值范围2的n次幂,n=0,1,2…15 */
BTIM_TimeBaseInit(CW_BTIM1, &BTIM_TimeBaseInitStruct);
//BTIM基本定时器初始化
    BTIM_ITConfig(CW_BTIM1, BTIM_IT_OV, ENABLE);   // 使能BTIMx的溢出中断
    BTIM_Cmd(CW_BTIM1, ENABLE);
    /*BTIM1以1MHz时钟进行计数,设置ARR寄存器为49999,则BTIM1将每50ms溢出一次,并触发中断服务程序。*/
    while (1)
    {
        /* 中断服务程序中 PB8输出翻转 */
    }
}

5.中断函数:PB08输出翻转

void BTIM1_IRQHandler(void)
{
    /* USER CODE BEGIN */
if (BTIM_GetITStatus(CW_BTIM1, BTIM_IT_OV))
// 检查BTIM的状态寄存器的各状态位是否置位
    {
        BTIM_ClearITPendingBit(CW_BTIM1, BTIM_IT_OV);  
        // 清除BTIM的状态寄存器的各状态位
        PB08_TOG();
//PB08口信号翻转
    }

    /* USER CODE END */
}

6.实验验证

示波器检测PB08口的信号输出,示波器波形图呈周期性翻转。

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是使用GD32高级定时器中断控制BLDC的基本代码框架。在这个例子中,我们使用了TIM1来控制BLDC的三相电机。 首先,需要在初始化函数中配置TIM1高级定时器,将它配置为PWM模式,并且设置计数器自动重载值。然后,配置TIM1的三个通道,使它们输出PWM波形。 然后,我们需要定义一个中断处理函数,在中断处理函数中,我们将更新三个PWM通道的占空比,以控制BLDC电机的转速。同时,我们还需要处理BLDC电机的正反转和停止操作。 最后,在主函数中,我们可以调用一个函数,使BLDC电机开始旋转。在这个例子中,我们使用了无定时器的方式来控制BLDC电机的转速,因此可以通过改变PWM占空比来改变电机的转速。 ``` #include "gd32f30x.h" #define PWM_FREQ 20000 // PWM频率 #define PWM_DUTY_CYCLE 50 // PWM占空比 // BLDC电机状态 enum { STOP, CW, CCW } bldc_state = STOP; // BLDC电机相序表 const uint8_t bldc_phases[6] = {1, 5, 4, 6, 2, 3}; // BLDC电机相序 uint8_t bldc_phase = 0; // 中断处理函数 void TIMER1_IRQHandler(void) { // 清除中断标志位 timer_interrupt_flag_clear(TIMER1, TIMER_INT_FLAG_CH0 | TIMER_INT_FLAG_CH1 | TIMER_INT_FLAG_CH2); // 更新PWM占空比 if (bldc_state == CW) { timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_0, PWM_DUTY_CYCLE * 10); timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_1, 0); timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_2, 0); } else if (bldc_state == CCW) { timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_0, 0); timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_1, PWM_DUTY_CYCLE * 10); timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_2, 0); } else { timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_0, 0); timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_1, 0); timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_2, 0); } // 更新BLDC电机相序 if (bldc_phase < 6) { timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_3, PWM_DUTY_CYCLE * bldc_phases[bldc_phase]); } else { timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_3, 0); } bldc_phase++; if (bldc_phase > 11) { bldc_phase = 0; } } // 初始化函数 void init(void) { // 使能GPIO和TIMER1时钟 rcu_periph_clock_enable(RCU_GPIOA); rcu_periph_clock_enable(RCU_TIMER1); // 配置GPIO gpio_init(GPIOA, GPIO_MODE_AF_PP, GPIO_OSPEED_50MHZ, GPIO_PIN_8 | GPIO_PIN_9 | GPIO_PIN_10 | GPIO_PIN_11); // 配置TIMER1 timer_deinit(TIMER1); timer_oc_parameter_struct timer_ocinitpara; timer_parameter_struct timer_initpara; timer_struct_para_init(&timer_initpara); timer_initpara.prescaler = SystemCoreClock / (PWM_FREQ * 1000) - 1; timer_initpara.alignedmode = TIMER_COUNTER_EDGE_ALIGNED_PWM_MODE; timer_initpara.counterdirection = TIMER_COUNTER_UP; timer_initpara.period = 100 - 1; timer_initpara.clockdivision = TIMER_CKDIV_DIV1; timer_init(TIMER1, &timer_initpara); timer_struct_para_init(&timer_ocinitpara); timer_ocinitpara.ocpolarity = TIMER_OC_POLARITY_HIGH; timer_ocinitpara.ocnpolarity = TIMER_OCN_POLARITY_HIGH; timer_ocinitpara.outputstate = TIMER_CCX_ENABLE; timer_ocinitpara.outputnstate = TIMER_CCXN_DISABLE; timer_ocinitpara.ocidlestate = TIMER_OC_IDLE_STATE_LOW; timer_ocinitpara.ocnidlestate = TIMER_OCN_IDLE_STATE_LOW; timer_channel_output_config(TIMER1, TIMER_CH_0, &timer_ocinitpara); timer_channel_output_config(TIMER1, TIMER_CH_1, &timer_ocinitpara); timer_channel_output_config(TIMER1, TIMER_CH_2, &timer_ocinitpara); timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_0, 0); timer_channel_output_mode_config(TIMER1, TIMER_CH_0, TIMER_OC_MODE_PWM0); timer_channel_output_shadow_config(TIMER1, TIMER_CH_0, TIMER_OC_SHADOW_DISABLE); timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_1, 0); timer_channel_output_mode_config(TIMER1, TIMER_CH_1, TIMER_OC_MODE_PWM0); timer_channel_output_shadow_config(TIMER1, TIMER_CH_1, TIMER_OC_SHADOW_DISABLE); timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_2, 0); timer_channel_output_mode_config(TIMER1, TIMER_CH_2, TIMER_OC_MODE_PWM0); timer_channel_output_shadow_config(TIMER1, TIMER_CH_2, TIMER_OC_SHADOW_DISABLE); timer_channel_output_config(TIMER1, TIMER_CH_3, &timer_ocinitpara); timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_3, 0); timer_channel_output_mode_config(TIMER1, TIMER_CH_3, TIMER_OC_MODE_PWM0); timer_channel_output_shadow_config(TIMER1, TIMER_CH_3, TIMER_OC_SHADOW_DISABLE); // 配置中断 nvic_irq_enable(TIMER1_IRQn, 0, 0); timer_interrupt_enable(TIMER1, TIMER_INT_CH0 | TIMER_INT_CH1 | TIMER_INT_CH2); // 启动TIMER1 timer_enable(TIMER1); } // BLDC电机启动函数 void bldc_start(void) { bldc_state = CW; } // BLDC电机停止函数 void bldc_stop(void) { bldc_state = STOP; bldc_phase = 0; timer_channel_output_pulse_value_config(TIMER1, TIMER_CH_3, 0); } // BLDC电机反转函数 void bldc_reverse(void) { if (bldc_state == CW) { bldc_state = CCW; } else if (bldc_state == CCW) { bldc_state = CW; } } int main(void) { init(); bldc_start(); while (1) { // 等待中断 } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值