一、实验概述

以下说明描述三种改变PWM占空比的方式,对于当前PWM输出波形的影响

1、禁止预装载功能,在PWM某一周期波形输出过程中改变占空比值(ccp)

2、禁止预装载功能,在PWM某周期波形输出完成后改变占空比值(ccp)

3、使能预装载功能,在PWM某一周期波形输出过程中改变占空比值(ccp)

4、使能预装载功能,在PWM某周期波形输出完成后改变占空比值(ccp)

以沁恒CH32V307VCT6芯片测试为基准,测试开发板:CH32V307V-R1-1V0

测试代码:见文章末尾

二、预装载功能解析

  • 代码解析

TIM_OC1PreloadConfig ( TIM1, TIM_OCPreload_Enable );

void TIM_OC1PreloadConfig(TIM_TypeDef *TIMx, uint16_t TIM_OCPreload)

{

uint16_t tmpccmr1 = 0;

tmpccmr1 = TIMx->CHCTLR1;

tmpccmr1 &= (uint16_t) ~((uint16_t)TIM_OC1PE);

tmpccmr1 |= TIM_OCPreload;

TIMx->CHCTLR1 = tmpccmr1;

}

#define TIM_OC1PE ((uint16_t)0x0008) /* Output Compare 1 Preload enable */

解析:从代码中可以看出底层代码是对“输出比较寄存器1(TIMx_CHCTLR1)”的

bit3位进行置位和初始化;

  • 手册关于此位功能解析

“Bit3:OC1PE-输出比较寄存器1预装载使能位”详细描述如下:

输出比较寄存器1预装载使能位

1:开启输出比较寄存器1(TIMx_CH1CVR)的预装载功能,读写操作仅对预装载寄存器操作,输出比较寄存器1的预装载值在更新事件到来时被加载至当前影子寄存器中;

0:禁止输出比较寄存器1的预装载功能,可随时写入输出比较寄存器1,并且新写入的数值立即起作用

  • 预装载功能详细解析
  1. 首先我们所用的“输出比较寄存器1(TIMx_CH1CVR)”实际上均是由两个寄存器组成的:Preload register(预装载寄存器)+ Shadow register(影子寄存器)
  2. 如果OC1PE=0,即不使能CH1CVR的预装载功能,则修改TIMx_CH1CVR寄存器的值就是操作影子寄存器,新的CH1CVR的值会立即生效。
  3. 如果OC1PE=1,即使能CH1CVR的预装载功能,则修改TIMx_CH1CVR 寄存器的值就是操作预装载寄存器,要等到发生更新事件后,TIMx_CH1CVR预装载寄存器的值才会拷贝到影子寄存器中,进而新的CH1CVR的值才会生效。

三、实验代码解析如下

设定系统时钟为SYSCLK_CLOCK=48MHz;

设定定时器1预分频系数PSC=48000-1,即定时器1ms记一个数;

设定定时器1自动重装载值ARR=100-1,即定时器计数到100更新一次,周期为100ms;

设定TIM1_CH1、TIM1_CH2、TIM1_CH3、TIM1_CH4均为输出比较模式,且配置相同。

设定定时器1输出比较寄存器1、2、3、4的值为CH1CVR=CH2CVR=CH3CVR= CH4CVR =CCP=30。

禁止输出比较寄存器1(TIM1_CH1)的预装载功能

使能输出比较寄存器2(TIM1_CH2)的预装载功能。

禁止输出比较寄存器3(TIM1_CH3)的预装载功能。

使能输出比较寄存器4(TIM1_CH4)的预装载功能。

开启定时器1更新中断,500ms后在中断中改变CH1CVR=CH2CVR=90

设定定时器2,每51ms进一次中断,在459ms后在中断中改变CH3CVR=CH4CVR=90

四个通道波形如下所示:

波形图解析:

  • 对于TIM1_CH1,禁止输出比较寄存器1(TIM1_CH1)的预装载功能;

在波形图箭头1处TIM1->CH1CVR=90;的操作相当于将值直接存入影子寄存器(波形图箭头1处),新的CH2CVR的值会立刻生效。

  • 对于TIM1_CH2,开启输出比较寄存器2(TIM1_CH2)的预装载功能

在波形图箭头1处TIM1->CH2CVR=90;的操作相当于将值存入预装载寄存器;

在等待发生更新事件(波形图箭头2处)后,会将值拷贝到影子寄存器;

进而新的CH1CVR的值才会生效。

  • 对于TIM1_CH3,禁止输出比较寄存器3(TIM1_CH3)的预装载功能;

在波形图箭头3处TIM1->CH1CVR=90;的操作相当于将值直接存入影子寄存器(波形图箭头3处),新的CH3CVR的值会立刻生效。由于此时的计数值小于90,故而电平会发生翻转为高电平。

  • 对于TIM1_CH4,开启输出比较寄存器4(TIM1_CH4)的预装载功能;

在波形图箭头3处TIM1->CH4CVR=90;的操作相当于将值存入预装载寄存器;

在等待发生更新事件(波形图箭头4处)后,会将值拷贝到影子寄存器;进而新的CH4CVR的值才会生效。

实验代码如下:

#include "debug.h"

void TIM1_UP_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));

void TIM2_IRQHandler(void) __attribute__((interrupt("WCH-Interrupt-fast")));

u16 i=0;

u16 j=0;

void TIM1_OutCompare_Init( u16 arr, u16 psc, u16 ccp )

{

GPIO_InitTypeDef GPIO_InitStructure={0};

TIM_OCInitTypeDef TIM_OCInitStructure={0};

TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure={0};

NVIC_InitTypeDef NVIC_InitStructure = {0};

RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOA | RCC_APB2Periph_TIM1, ENABLE );

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init( GPIOA, &GPIO_InitStructure );

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init( GPIOA, &GPIO_InitStructure );

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init( GPIOA, &GPIO_InitStructure );

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

GPIO_Init( GPIOA, &GPIO_InitStructure );

TIM_TimeBaseInitStructure.TIM_Period = arr;

TIM_TimeBaseInitStructure.TIM_Prescaler = psc;

TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;

TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;

TIM_TimeBaseInit( TIM1, &TIM_TimeBaseInitStructure);

TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;

TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;

TIM_OCInitStructure.TIM_Pulse = ccp;

TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;

TIM_OC1Init( TIM1, &TIM_OCInitStructure );

TIM_OC2Init( TIM1, &TIM_OCInitStructure );

TIM_OC3Init( TIM1, &TIM_OCInitStructure );

TIM_OC4Init( TIM1, &TIM_OCInitStructure );

NVIC_InitStructure.NVIC_IRQChannel = TIM1_UP_IRQn;

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

NVIC_Init(&NVIC_InitStructure);

TIM_CtrlPWMOutputs(TIM1, ENABLE );

TIM_OC1PreloadConfig( TIM1, TIM_OCPreload_Disable );

TIM_OC2PreloadConfig( TIM1, TIM_OCPreload_Enable );

TIM_OC3PreloadConfig( TIM1, TIM_OCPreload_Disable );

TIM_OC4PreloadConfig( TIM1, TIM_OCPreload_Enable );

TIM_ARRPreloadConfig( TIM1, ENABLE );

TIM_ClearITPendingBit(TIM1,TIM_IT_Update);

TIM_ITConfig(TIM1,TIM_IT_Update,ENABLE);

TIM_Cmd( TIM1, ENABLE );

}

void TIM2_CNT_Init(u16 arr, u16 psc)

{

TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure={0};

NVIC_InitTypeDef NVIC_InitStructure = {0};

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE );

TIM_TimeBaseInitStructure.TIM_Period = arr;

TIM_TimeBaseInitStructure.TIM_Prescaler = psc;

TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;

TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;

TIM_TimeBaseInit( TIM2, &TIM_TimeBaseInitStructure);

NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

NVIC_Init(&NVIC_InitStructure);

TIM_ARRPreloadConfig( TIM2, ENABLE );

TIM_ClearITPendingBit(TIM2,TIM_IT_Update);

TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE);

TIM_Cmd( TIM2, ENABLE );

}

void TIM1_UP_IRQHandler(void)

{

if(TIM_GetITStatus(TIM1,TIM_IT_Update) == SET)

{

i++;

if(i==5)

{

TIM1->CH1CVR=90;

TIM1->CH2CVR=90;

}

TIM_ClearFlag(TIM1,TIM_IT_Update);

}

}

void TIM2_IRQHandler(void)

{

if(TIM_GetITStatus(TIM2,TIM_IT_Update) == SET)

{

j++;

if(j==9)

{

TIM1->CH3CVR=90;

TIM1->CH4CVR=90;

}

TIM_ClearFlag(TIM2,TIM_IT_Update);

}

}

int main(void)

{

TIM1_OutCompare_Init( 100-1, 48000-1, 30 );//1ms记一个数

TIM2_CNT_Init(51-1, 48000-1);//51ms

while(1);

}

TIM-PWM输出,占空比改变时机对输出波形的影响的更多相关文章

  1. STM32 HAL库学习系列第4篇 定时器TIM----- 开始定时器与PWM输出配置

    基本流程: 1.配置定时器 2.开启定时器 3.动态改变pwm输出,改变值  HAL_TIM_PWM_Start(&htim4, TIM_CHANNEL_1); 函数总结: __HAL_TIM ...

  2. NUCLEO-L053R8 TIM定时器 PWM输出

    TIM2 PWM输出测试 今天给大伙分享一个TIM2 PWM输出小实验. 实验开发板:Nucleo-L053R8,即STM32L053R8T6. 开发环境:MDK5 图1 - 工程界面 本次实验测试的 ...

  3. (五)转载:通用定时器PWM输出

    1.     TIMER输出PWM基本概念 脉冲宽度调制(PWM),是英文“Pulse Width Modulation”的缩写,简称脉宽调制,是利用微处理器的数字输出来对模拟电路进行控制的一种非常有 ...

  4. STM32(7)——通用定时器PWM输出

    1.TIMER输出PWM基本概念         脉冲宽度调制(PWM),是英文“Pulse Width Modulation”的缩写,简称脉宽调制,是利用微处理器的数字输出来对模拟电路进行控制的一种 ...

  5. 16路PWM输出的pca9685模块

    今天要介绍的就是该模块,该模块是16路pwm模块,使用I2C总线可以控制16路舵机(led). 接线OE空着就可以,其他VCC是芯片供电+5,SCL时钟线,SDA信号线,GND地线. 芯片介绍可以看: ...

  6. 关于STM32CubeMX使用LL库设置PWM输出

    HAL和LL库 HAL是ST为了实现代码在ST家族的MCU上的移植性,推出的一个库,称为硬件抽象层,很明显,这样做将会牺牲存储资源,所以项目最后的代码比较冗余,且运行效率大大降低,运行速度受制于fla ...

  7. TIMER门控模式控制PWM输出长度

    TIMER门控模式控制PWM输出长度 参照一些网友代码做了些修改,由TIM4来控制TIM2的PWM输出长度, 采用主从的门控模式,即TIM4输出高时候TIM2使能输出 //TIM2 PWM输出,由TI ...

  8. STM32: TIMER门控模式控制PWM输出长度

    搞了两天单脉冲没搞定,无意中发现,这个利用主从模式的门控方式来控制一路PWM的输出长度很有效. //TIM2 PWM输出,由TIM4来控制其输出与停止 //frequency_tim2:TIM2 PW ...

  9. STM32F103ZET6 PWM输出

    1.通用定时器的PWM功能 STM32F103ZET6有4个通用定时器,分别是TIM2.TIM3.TIM4.TIM5. 通用定时器由一个可编程预分频器驱动的16位自动装载计数器构成. 通用定时器的很多 ...

  10. 合宙AIR105(三): 定时器, 定时器中断和PWM输出

    目录 合宙AIR105(一): Keil MDK开发环境, DAP-Link 烧录和调试 合宙AIR105(二): 时钟设置和延迟函数 合宙AIR105(三): 定时器, 定时器中断和PWM输出 Ai ...

随机推荐

  1. 《视觉SLAM十四讲》visual studio 19 + PCL点云创建图像与现实

    SLCM真是博大精深.之前简单的学习了OpenCV,主要是是使用python语言,现在学习SLAM需要使用C++,略难,但比起SLAM本身,不值一提. <视觉SLAM十四讲>里面的环境主要 ...

  2. Mysql5.7的安装与卸载与数据迁移

    Mysql5.7的安装与卸载 安装: 1.安装的时候 只选择安装 server即可 2.安装过程中,一定要重新选择安装路径和各个日志的路径,将来会非常大,不适合存在系统盘下面 3.安装时,取消MySQ ...

  3. store数据仓库

    项目搭建 npm init vite-app GxShujukucd GxShujukunpm inpm i vue-router npm i vuex // 这一句是这节课的关键 新建store ① ...

  4. 【转载】PostgreSQL逻辑订阅logical

    原文地址:https://blog.csdn.net/gguxxing008/article/details/106356086 逻辑订阅是PostgreSQL10.0开始支持的新功能,Postgre ...

  5. FTP文件夹错误:【打开FTP服务器上的文件夹时发生错误。请检查是否有权限访问该文件夹】

    资源管理器访问FTP服务器报错,提示FTP文件夹错误:[打开FTP服务器上的文件夹时发生错误.请检查是否有权限访问该文件夹]. 详细信息: 200 Switching to ASCII mode. 2 ...

  6. Word 找不到 Endnote选项

    Word 2010 找不到 Endnote选项汇总(不是Office有效加载项)因为基本百度上的问题我全都遇到了-说明:在我们使用Word的过程中,常常发现没有Endnote选项.然后去找百度方法:1 ...

  7. 程序猿、IT男、屌丝

    一个学计算机的,做了金融圈的朋友,对另外做硬件的朋友,如是说: [据说你写的程序代码,必须很浪的计算机才能运行] [ 以后还用写程序吗.不是ChatGPT直接可以出源代码吗?] 程序猿.IT男.屌丝. ...

  8. ClassLoader 双亲委派

    一个程序有一个默认的appClassLoader.类不是由被调用者也不是被自身加载的,正常情况下是被默认的AppClassLoader加载的. System.out.println(test3.cla ...

  9. Android笔记--事务处理+数据库版本升级

    事务处理 beginTransaction:开始事务的标志 setTransactionSuccessful:事务成功的标志 endTransaction:结束事务的标志 在上面的图片里面,首先进行事 ...

  10. 在java中String类为什么要设计成final?Java面试常见问题

    2023Java面试题最经典的问题之一了,非常经典的Java基础知识,一定要学会! 在Java中,String类被设计成final,这意味着它的值在创建后不可更改.这是因为字符串在Java中使用广泛, ...