STM32F103RCT6开发板M3单片机教程07-TIMER1CH1输出 PWM做LED呼吸灯

news2025/1/8 5:38:59

概述

本教程使用是(光明谷SUN_STM32mini开发板
 



免费开发板
 

在谷动谷力社区注册用户,打卡,发帖求助都可以获取积分,当然最主要是发原创应用文档奖励更多积分.
(可用积分换取,真的不用钱,开发板免费玩):STM32F103RCT6开发板M3单片机核芯小系统板学习板 ...
已经购买用的,可以注册用户,打卡,发帖求助都可以获取积分,当然最主要是发原创应用文档奖励更多积分.用积分换钱,返还您的买开发板钱
详细参阅
[推荐]STM32F103RCT6开发板M3单片机最小系统板学习板 带USB JTAG...

 
资料下载
STM32F103RCT6_SDK开发资料应用手册下载


书接上回:
上回我们用Timer2做了一个定时中断,让LED灯闪烁(STM32F103RCT6开发板M3单片机教程06--定时器中断 .. )

这回我们用Timer1的CH1 输出PWM做呼吸灯参看原理图(详细到 STM32F103RCT6_SDK开发资料应用手册下载
 


程序设计

何为呼吸灯?

慢慢变亮又慢慢变暗到灭,如此往复循环灯为呼吸灯也。

硬件解读

上回书得知PA8连接LED阴极,低电平点亮。
参考STM32F103RCT6数据手册, 发现PA8是Timer1的CH1,可以输出PWM
 



上回有云:
大容量的STM32F103xx增强型系列产品包含最多2个高级控制定时器、 4个普通定时器和2个基本定时器,以及2个看门狗定时器和1个系统嘀嗒定时器。
下表比较了高级控制定时器、普通定时器和基本定时器的功能:
 

高级控制定时器(TIM1和TIM8)

两个高级控制定时器(TIM1TIM8)可以被看成是分配到6个通道的三相PWM发生器,它具有带死区插入的互补PWM输出,还可以被当成完整的通用定时器。四个独立的通道可以用于:
● 输入捕获
● 输出比较
● 产生PWM(边缘或中心对齐模式)
● 单脉冲输出
配置为16位标准定时器时,它与TIMx定时器具有相同的功能。配置为16PWM发生器时,它具有全调制能力(0~100%)
在调试模式下,计数器可以被冻结,同时PWM输出被禁止,从而切断由这些输出所控制的开关。
很多功能都与标准的TIM定时器相同,内部结构也相同,因此高级控制定时器可以通过定时器链接功能与TIM定时器协同操作,提供同步或事件链接功能。


程序流程

1、系统初始化(RCC, GPIO)
2、设定初始LED亮度为0,亮度递增
3、高度到100%(全亮时),亮度递减
返回第2步



开发码码

复制上节的工程源,更改文件夹名
 

GPIO配置--PWM输出模式

修改BSP下Led.c, LED_GPIO_Config函数
更改主GPIO输出模式
原://        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;        //设置引脚工作模式为通用推挽输出
改:GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;     //PWMout

/*******************************************************************************
* 函数名  : LED_GPIO_Config
* 描述    : LED IO配置
* 输入    : 无
* 输出    : 无
* 返回    : 无
* 说明    : LED(1~4)的IO口分别是:PB5,PB6,PB7,PB8
*******************************************************************************/
void LED_GPIO_Config(void)
{
        GPIO_InitTypeDef  GPIO_InitStructure;                                //定义一个GPIO_InitTypeDef类型的GPIO初始化结构体
        
        RCC_APB2PeriphClockCmd(LED_RCC, ENABLE);                        //使能GPIOB的外设时钟        
        
        GPIO_InitStructure.GPIO_Pin = LED_ALL;                                //选择要初始化的GPIOB引脚(PA5,PA6,PA7,PA8)
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;     //PWMout
//        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;        //设置引脚工作模式为通用推挽输出                 
        GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;        //设置引脚输出最大速率为50MHz
        GPIO_Init(LED_PORT, &GPIO_InitStructure);                        //调用库函数中的GPIO初始化函数,初始化GPIOB中的PA5,PA6,PA7,PA8引脚
   

        LED_ALL_OFF();                                                                                //关闭ALL_LED                                          
}


定时器配置--PWM模式

修改main.c,  原Timer2定时函数, 修改通用Timer_Init_Config定时中断配置函数(这个也可以不修改不影响功能,为了让源码更接近实际项目代码,提高代码可用性,可移植性,所以这样修改)一般来说TimerPeriod越大越好越定时越精确,但不能超过16bits 最大数65535。

/*******************************************************************************
* 函数名  : Timer_Init_Config
* 描述    : Timer初始化配置
* 输入    : TIM_TypeDef* TIMx 定时器名 TIM1~8
            uint16_t fre_hz 定时器 中断 频率
* 输出    : 无
* 返回    : 无
* 说明    : 无
*******************************************************************************/
void Timer_Init_Config(TIM_TypeDef* TIMx, uint16_t fre_hz)
{
        TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
        NVIC_InitTypeDef NVIC_InitStructure;
        uint16_t TimerPeriod = 0;
    uint16_t TimerPrescaler = 71; //1Mhz Timer Conter frequency
   
    /* Compute the value to be set in ARR regiter to generate signal frequency at Fre_hz */
        TimerPeriod = (SystemCoreClock / fre_hz /( TimerPrescaler +1)) - 1;
        TIM_TimeBaseStructure.TIM_Period = TimerPeriod; //4999;                                        //设置在下一个更新事件装入活动的自动重装载寄存器周期的值(计数到5000为500ms)
        TIM_TimeBaseStructure.TIM_Prescaler = TimerPrescaler; //7199;                                        //设置用来作为TIMx时钟频率除数的预分频值(10KHz的计数频率) 71
        TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;                //设置时钟分割:TDTS = TIM_CKD_DIV1
        TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;        //TIM向上计数模式
        TIM_TimeBaseInit(TIMx, &TIM_TimeBaseStructure);                                //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位

        /*中断优先级NVIC设置*/
        NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;                                //TIM2中断
        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;        //先占优先级1级
        NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;                        //从优先级1级
        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;                                //使能IRQ通道
        NVIC_Init(&NVIC_InitStructure);                                                         //初始化NVIC寄存器
         
        TIM_ITConfig(TIMx, TIM_IT_Update, ENABLE );                                 //使能TIMx指定的中断
        
        TIM_Cmd(TIMx, ENABLE);                                                                          //使能TIMx外设
}         



修改main函数

/*******************************************************************************
* 函数名  : main
* 描述    : 主函数,用户程序从main函数开始运行
* 输入    : 无
* 输出    : 无
* 返回值  : int:返回值为一个16位整形数
* 说明    : 无
*******************************************************************************/
int main(void)
{
    u8 keyVal;
    RCC_Configuration();
    SysTick_Init_Config();
        USART1_Init_Config(115200);//USART1初始化配置
    LED_GPIO_Config();
    Key_GPIO_Config();
   
    printf ("*===================================================*\n");
    printf ("*  *  Name: Sun STM32 mini Demo Code.    *************\n");
    printf ("*  * (C) Sunshine Silicon Corporation    *************\n");
    printf ("*  *  Website: http://www.sunsili.com    *************\n");
    printf ("*  *   E-Mail : fan@sunsili.com          *************\n");
    printf ("*===================================================*\n");
    printf ("* Sun STM32 mini PWM  Demo code .*\n");
   
    lum = 0;    //Start luminance 0
    flag = 1;   //Up count in
    Timer_Ch1_pwm_init(TIM1, 10000, lum);
    Timer_Init_Config(TIM2, 100);        //Timer2初始化配置 100Hz 中断频率 10ms中断一次
   
        while (1)
        {
        if(tim2_tick)
        {
            tim2_tick = 0;
            if(flag)
            {
               if(++lum >= 100)
               {
                   flag = 0;    //Down count in
               }
//               LED4_ON();
            }
            else
            {               
                if(!lum)
                {
                    flag = 1;
                }
                else
                {
                    lum--;
                }
//                LED4_OFF();
            }
            TIM_OC1_set_Pulse(TIM1, 10000, 100-lum);
        }
}


增加:

/*******************************************************************************
* 函数名  : Timer_Ch1_pwminit
* 描述    : 定时器 Ch1 pwm 初始化
* 输入    : TIM_TypeDef* TIMx  定时器名 TIM1~8
            uint16_t fre_hz PWM 频率
            uint16_t dty 占空比 0-100
* 输出    : 无
* 返回    : 无
* 说明    : 无
*******************************************************************************/
void Timer_Ch1_pwm_init(TIM_TypeDef* TIMx, uint16_t fre_hz, uint16_t dty)
{
    TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
    TIM_OCInitTypeDef  TIM_OCInitStructure;
    uint16_t TimerPeriod = 0;
    uint16_t Channel1Pulse = 0;
   
    /* TIM1 Configuration ---------------------------------------------------
   Generate 7 PWM signals with 4 different duty cycles:
   TIM1CLK = SystemCoreClock, Prescaler = 0, TIM1 counter clock = SystemCoreClock
   SystemCoreClock is set to 72 MHz for Low-density, Medium-density, High-density
   and Connectivity line devices and to 24 MHz for Low-Density Value line and
   Medium-Density Value line devices
   
   The objective is to generate 7 PWM signal at 17.57 KHz:
     - TIM1_Period = (SystemCoreClock / 17570) - 1
   The channel 1 and channel 1N duty cycle is set to 50%
   The channel 2 and channel 2N duty cycle is set to 37.5%
   The channel 3 and channel 3N duty cycle is set to 25%
   The channel 4 duty cycle is set to 12.5%
   The Timer pulse is calculated as follows:
     - ChannelxPulse = DutyCycle * (TIM1_Period - 1) / 100
  ----------------------------------------------------------------------- */
  /* Compute the value to be set in ARR regiter to generate signal frequency at Fre_hz */
  TimerPeriod = (SystemCoreClock / fre_hz ) - 1;
  /* Compute CCR1 value to generate a duty cycle at duty for channel 1 and 1N */
  Channel1Pulse = (uint16_t) (((uint32_t) dty * (TimerPeriod - 1)) / 100);
  /* Compute CCR2 value to generate a duty cycle at 37.5%  for channel 2 and 2N */
//  Channel2Pulse = (uint16_t) (((uint32_t) 375 * (TimerPeriod - 1)) / 1000);
  /* Compute CCR3 value to generate a duty cycle at 25%  for channel 3 and 3N */
//  Channel3Pulse = (uint16_t) (((uint32_t) 25 * (TimerPeriod - 1)) / 100);
  /* Compute CCR4 value to generate a duty cycle at 12.5%  for channel 4 */
//  Channel4Pulse = (uint16_t) (((uint32_t) 125 * (TimerPeriod- 1)) / 1000);

  /* Time Base configuration */
  TIM_TimeBaseStructure.TIM_Prescaler = 0;
  TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
  TIM_TimeBaseStructure.TIM_Period = TimerPeriod;
  TIM_TimeBaseStructure.TIM_ClockDivision = 0;
  TIM_TimeBaseStructure.TIM_RepetitionCounter = 0;

  TIM_TimeBaseInit(TIM1, &TIM_TimeBaseStructure);

  /* Channel 1, 2,3 and 4 Configuration in PWM mode */
  TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM2;
  TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
  TIM_OCInitStructure.TIM_OutputNState = TIM_OutputNState_Enable;
  TIM_OCInitStructure.TIM_Pulse = Channel1Pulse;
  TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_Low;
  TIM_OCInitStructure.TIM_OCNPolarity = TIM_OCNPolarity_High;
  TIM_OCInitStructure.TIM_OCIdleState = TIM_OCIdleState_Set;
  TIM_OCInitStructure.TIM_OCNIdleState = TIM_OCIdleState_Reset;

  TIM_OC1Init(TIMx, &TIM_OCInitStructure);
  
  /* TIM1 counter enable */
  TIM_Cmd(TIMx, ENABLE);

  /* TIM1 Main Output Enable */
  TIM_CtrlPWMOutputs(TIMx, ENABLE);

}


/*******************************************************************************
* 函数名  : TIM_OC1_set_Pulse
* 描述    : 定时器 Ch1 pwm 脉冲相关设置
* 输入    : TIM_TypeDef* TIMx  定时器名 TIM1~8
            uint16_t prd  PWD周期
            uint16_t dty 占空比 0-100
* 输出    : 无
* 返回    : 无
* 说明    : 无
*******************************************************************************/
void TIM_OC1_set_Pulse(TIM_TypeDef* TIMx, uint16_t fre_hz, uint16_t dty)
{
    uint16_t Channel1Pulse = 0;
    uint16_t TimerPeriod = 0;
   
     /* Compute the value to be set in ARR regiter to generate signal frequency at Fre_hz */
    TimerPeriod = (SystemCoreClock / fre_hz ) - 1;
    /* Compute CCR1 value to generate a duty cycle at duty for channel 1 and 1N */
    Channel1Pulse = (uint16_t) (((uint32_t) dty * (TimerPeriod - 1)) / 100);
    /* Set the Capture Compare Register value */
    TIMx->CCR1 = Channel1Pulse;
}


修改:RCC_Configuration函数

/*******************************************************************************
* 函数名  : RCC_Configuration
* 描述    : 设置系统时钟为72MHZ(这个可以根据需要改)
* 输入    : 无
* 输出    : 无
* 返回值  : 无
* 说明    : STM32F107x和STM32F105x系列MCU与STM32F103x系列MCU时钟配置有所不同
*******************************************************************************/
void RCC_Configuration(void)
{
  ErrorStatus HSEStartUpStatus;               //外部高速时钟(HSE)的工作状态变量
  
  RCC_DeInit();                               //将所有与时钟相关的寄存器设置为默认值
  RCC_HSEConfig(RCC_HSE_ON);                  //启动外部高速时钟HSE
  HSEStartUpStatus = RCC_WaitForHSEStartUp(); //等待外部高速时钟(HSE)稳定

  if(SUCCESS == HSEStartUpStatus)             //如果外部高速时钟已经稳定
  {
    /* Enable Prefetch Buffer */
    FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); //Flash设置
    /* Flash 2 wait state */
    FLASH_SetLatency(FLASH_Latency_2);
   
  
    RCC_HCLKConfig(RCC_SYSCLK_Div1); //设置AHB时钟等于系统时钟(1分频)/72MHZ
    RCC_PCLK2Config(RCC_HCLK_Div1);  //设置APB2时钟和HCLK时钟相等/72MHz(最大为72MHz)
    RCC_PCLK1Config(RCC_HCLK_Div2);  //设置APB1时钟是HCLK时钟的2分频/36MHz(最大为36MHz)
   
    RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9); //PLLCLK = 8MHz * 9 = 72 MHz


    RCC_PLLCmd(ENABLE); //使能PLL
    while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET); //等待PLL稳定

    RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);          //设置系统时钟的时钟源为PLL

    while(RCC_GetSYSCLKSource() != 0x08);               //检查系统的时钟源是否是PLL
    RCC_ClockSecuritySystemCmd(ENABLE);                 //使能系统安全时钟
   
    /* TIM1 GPIOx clock enable */
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1 | RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB | RCC_APB2Periph_GPIOC | RCC_APB2Periph_GPIOD | RCC_APB2Periph_GPIOE | RCC_APB2Periph_AFIO, ENABLE);
    /* TIM2 clock enable */
    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);
  }
}

编译调试

编译


保存全部,编译, 下载。

调试

点运行后,观察运行结果 
运行结果:

串口打印日志信息

 慢慢变亮又慢慢变暗到灭,如此往复循环灯为呼吸灯也。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1377986.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Mysql InnoDB行锁深入理解

Record Lock记录锁 Record Lock 称为记录锁,锁住的是一条记录。而且记录锁是有 S 锁和 X 锁之分的: 当一个事务对一条记录加了 S 型记录锁后,其他事务也可以继续对该记录加 S 型记录锁(S 型与 S 锁兼容),…

NAND SCA接口对性能影响有多大?

在多LUN场景下,SCA接口尤其有助于提高随机读取性能。通过合理安排读取命令和等待时间(如tR),SCA接口可以在一个LUN完成读取后立即开始另一个LUN的读取操作,而无需等待整个DQ总线空闲,从而减少了延迟和提高了…

设计一个简易版的数据库路由

👏作者简介:大家好,我是爱吃芝士的土豆倪,24届校招生Java选手,很高兴认识大家📕系列专栏:Spring原理、JUC原理、Kafka原理、分布式技术原理、数据库技术🔥如果感觉博主的文章还不错的…

双指针问题——求只包含两个元素的最长连续子序列(子数组)

一,题目描述 你正在探访一家农场,农场从左到右种植了一排果树。这些树用一个整数数组 fruits 表示,其中 fruits[i] 是第 i 棵树上的水果 种类 。 你想要尽可能多地收集水果。然而,农场的主人设定了一些严格的规矩,你必…

vue前端开发自学,组件的生命周期函数介绍001

vue前端开发自学,组件的生命周期函数介绍001!今天介绍一下,组件自身的生命周期函数。又叫做,钩子函数。可以借助于这些钩子函数,实现很多我们预想的效果。比如,在组件渲染 之前,就做一些特殊的操作等等。 …

什么是 CAS

程序员的公众号:源1024,获取更多资料,无加密无套路! 最近整理了一波电子书籍资料,包含《Effective Java中文版 第2版》《深入JAVA虚拟机》,《重构改善既有代码设计》,《MySQL高性能-第3版》&…

Javaweb之SpringBootWeb案例查询部门以及前后端联调的详细解析

2.1 查询部门 2.1.1 原型和需求 查询的部门的信息:部门ID、部门名称、修改时间 通过页面原型以及需求描述,我们可以看到,部门查询,是不需要考虑分页操作的。 2.1.2 接口文档 部门列表查询 基本信息 请求路径:/depts …

内存卡为什么会提示格式化,内存卡提示格式化还能恢复吗

对于许多电脑用户来说,执行内存卡格式化操作导致数据丢失是一个常见的问题。在日常生活中,数据丢失的情况并不少见,但内存卡格式化后的数据恢复相对较难。目前,能够使用的方法较少,且成功率较低,但并不是没…

NAND系统性能提升常见方案

随着NAND的发展,针对NAND系统性能提升,业内目前主要的做法有以下几种方案: 1.提升总线频率和优化AC时序: 提高NAND闪存接口的工作频率可以显著加快数据传输速度。通过不断改进工艺和技术,缩短了信号稳定时间、降低了延…

Linux第29步_虚拟机连接(与主机断开连接)U盘选项为灰色解决方法

在WIN11中,虚拟机“连接(与主机断开连接)U盘”选项为灰色,解决方法如下: 1、关闭虚拟机电源,得到下面的界面: 2、根据上述提示,找到虚拟机所在磁盘 3、配置文件属性见下图: 4、使用记事本打开…

vim基本操作命令

一、vi简介 vi是“Visual interface”的简称,它在Linux上的地位就仿佛Edit程序在DOS上一样。它可以执行输出、删除、查找、替换、块操作等众多文本操作,而且用户可以根据自己的需要对其进行定制。Vi不是一个排版程序,它不象Word或WPS那样可以…

“具身智能”浪潮中,达闼机器人的商业化“奇点”已然到来?

当前,人形机器人产业正在快速发展,而2023年必将会是载入史册的一年。 具体来看,2023年,AI技术大爆发,可在语言、视觉、运动控制、降低研发成本等多方面赋能人形机器人产业发展。与此同时,特斯拉、波士顿动…

NAND Separate Command Address (SCA) 接口命令解读

CA output packet和CA input packet是Separate Command Address (SCA) NAND接口协议中用于命令和地址传输的关键数据结构。 CA Input Packet: 在SCA接口中,输入到NAND器件的命令和地址信息被组织成并行至串行转换的CA(Command and Address)输…

linux创建文件并分配权限

linux中对文件的定义 在Linux中,文件是一个具有符号名字的一组相关联元素的有序序列。文件可以包含的内容十分广泛,操作系统和用户都可以将具有一定独立功能的一个程序模块、一组数据或一组文字命名为一个文件。文件名是数据有序序列集合(文…

Asp .Net Core 系列: 集成 CORS跨域配置

文章目录 什么是CORS?Asp .Net Core 中如何配置CORS?CorsPolicyBuilder类详解注册以及使用策略三种方式EnableCors 和 DisableCors 特性关于带证书与不带证书代码的实现跨源(cross-origin)不带请求证书(Credentials)跨源(cross-origin&…

【国产之光】开年尝鲜——优秀的AI编码助手 Fitten Code

文章目录 前言1. 工具准备1.0 事先说明1.1 VSCode1.2 Fitten Code1.3 GitHub Copilot 2. 使用测评2.1 需求理解2.2 上下文理解 3. 总结推荐链接 开年尝鲜高质量国产AI编码助手——FittenCode 前言 2024年刚刚开局,清华大学 与 非十科技 就发布了全新的 VSCode AI…

H264码流进行RTP包封装

一.H264基本概念 H.264从框架结构上分为视频编码层(VCL)和网络抽象层(NAL),VCL功能是进行视频编解码,包括运动补偿预测,变换编码和熵编码等功能;NAL用于采用适当的格式对VCL视频数据…

CAN201计网历年大题收集

网络性能计算 e.g1.1 算RTT 传输时间 when you enter the URL in your browser, it initiates an HTTP request. This request is encapsulated in TCP, which is then further encapsulated in IP for routing. The data is transmitted over the physical medium using link …

中国信通院联合发布《数字孪生城市白皮书(2023年)》

2017年“数字孪生城市”概念被首次提出,2021年我国“十四五”规划纲要明确“探索建设数字孪生城市”,2023年《数字中国建设整体布局规划》再次提出“全面提升数字中国建设的整体性、系统性、协同性”以及“探索建设数字孪生城市”等要求。数字孪生城市建…

ES高级查询

ES中提供了一种强大的检索数据方式,这种检索方式称为Query DSL,这种方式的丰富查询语法让ES检索变得更强大,更简洁。 1.常见查询 1.1查询所有[match_all] match_all关键字:返回索引中的全部文档。 GET /products/_search { &…