STM32 PWM 详解(基于 STM32F429 HAL 库)

news2024/11/23 20:25:44

目录

前言

一、PWM 简介

二、STM32F429 的 PWM 功能

1.定时器资源

2.PWM 模式

3.PWM原理图

三、使用 HAL 库配置 STM32F429 的 PWM

1.开启时钟

2.配置定时器

3.配置通道 

4.启动定时器 

5.PWM 占空比的调节 

四、应用实例

五、总结


前言

        在嵌入式系统开发中,脉冲宽度调制(PWM)是一种常用的技术,它可以通过控制脉冲的宽度来调节输出信号的平均电压或功率。STM32 系列微控制器提供了强大的 PWM 功能,本文将详细介绍如何在 STM32F429 上使用 HAL 库实现 PWM 输出。

一、PWM 简介

        PWM 是一种数字信号调制技术,它通过在固定的周期内改变高电平和低电平的时间比例来实现对模拟信号的模拟。在 STM32 中,PWM 可以用于控制电机速度、LED 亮度、音频输出等。

二、STM32F429 的 PWM 功能

        STM32F429 微控制器具有多个定时器,其中一些定时器可以配置为 PWM 输出模式。这些定时器可以产生不同频率和占空比的 PWM 信号,以满足各种应用需求。

1.定时器资源

  • STM32F429 通常具有多个通用定时器(TIM2、TIM3、TIM4 等)和高级控制定时器(TIM1、TIM8 等),这些定时器都可以配置为 PWM 输出模式。
  • 每个定时器通常有多个通道,可以同时产生多个独立的 PWM 信号。

2.PWM 模式

  • 边沿对齐模式:在这种模式下,计数器从 0 开始计数,当计数值达到比较值时,输出电平翻转。可以设置为向上计数、向下计数或向上 / 向下计数模式。
  • 中心对齐模式:计数器从 0 开始计数,当计数值达到自动重载值的一半时,输出电平翻转,然后继续计数到自动重载值时,输出电平再次翻转并重新开始计数。

3.PWM原理图

PWM原理图如下图所示:

 如果定时器工作在向上计数 PWM模式,且当 CNT<CCRx 时,输出 0,当 CNT>=CCRx 时输出 1。那么就可以得到上面的 PWM示意图:当 CNT 值小于 CCRx 的时候,IO 输出低电平(0),当 CNT 值大于等于 CCRx 的时候,IO 输出高电平(1),当 CNT 达到 ARR 值的时候,重新归零,然后重新向上计数,这样依次进行循环。改变 CCRx 的值,就可以改变 PWM 输出的占空比,改变 ARR 的值,就可以改变 PWM 输出的频率

三、使用 HAL 库配置 STM32F429 的 PWM

1.开启时钟

在使用 PWM 功能之前,需要先开启相应的定时器时钟。可以使用 HAL_RCC_GetPeriphClkEnable () 函数获取时钟状态,并使用 HAL_RCC_EnablePeriphClk () 函数开启时钟。

2.配置定时器

使用 HAL_TIM_PWM_Init () 函数初始化定时器为 PWM 模式。需要设置定时器的预分频系数、自动重载值、计数模式等参数。

例如:

TIM_HandleTypeDef htim;
htim.Instance = TIMx;
htim.Init.Prescaler = prescaler;
htim.Init.Period = period;
htim.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
htim.Init.CounterMode = TIM_COUNTERMODE_UP;
HAL_TIM_PWM_Init(&htim);

3.配置通道 

使用 HAL_TIM_PWM_ConfigChannel () 函数配置定时器通道为 PWM 输出模式。需要设置通道的极性、比较值等参数。

例如:

TIM_OC_InitTypeDef sConfigOC;
sConfigOC.OCMode = TIM_OCMODE_PWM1;
sConfigOC.Pulse = pulse;
sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH;
sConfigOC.OCFastMode = TIM_OCFAST_DISABLE;
HAL_TIM_PWM_ConfigChannel(&htim, &sConfigOC, TIM_CHANNEL_x);

4.启动定时器 

使用 HAL_TIM_PWM_Start () 函数启动定时器,开始产生 PWM 信号。

例如:

HAL_TIM_PWM_Start(&htim, TIM_CHANNEL_x);

5.PWM 占空比的调节 

(1)什么是占空比

想象一下有一个小灯,我们希望通过一个特殊的开关来控制它的亮度。这个开关不是普通的开和关,而是可以快速地一开一关,开和关的速度非常快,快到我们的眼睛都察觉不出来它在闪烁。

PWM(脉冲宽度调制)就像是这个特殊的开关。占空比呢,就是在一个固定的时间段内,这个开关打开的时间占总时间的比例。

比如说,我们把这个时间段想象成一分钟。如果占空比是 50%,那就意味着在这一分钟里,开关有一半的时间是打开的,有一半的时间是关闭的。这样小灯在这一分钟里就会有一半的时间是亮着的,一半的时间是暗着的,整体看起来就像是亮度减半了。

如果占空比是 20%,那在这一分钟里,开关只有五分之一的时间是打开的,其余时间都是关闭的,小灯就会比较暗。而如果占空比是 80%,开关大部分时间都是打开的,小灯就会很亮。

所以,占空比决定了通过 PWM 控制的设备(比如小灯、电机等)的工作状态的强度。占空比越大,设备就越接近 “完全开启” 的状态;占空比越小,设备就越接近 “完全关闭” 的状态。

(2)如何调节占空比?

可以通过修改定时器通道的比较值来调节 PWM 的占空比。比较值越大,占空比越高

__HAL_TIM_SET_COMPARE(&htim, TIM_CHANNEL_x, new_pulse);

 

四、应用实例

以下是一个使用 STM32F429 HAL 库实现 PWM 控制 LED 亮度的示例代码:

 

#include "main.h"
#include "stm32f4xx_hal.h"

TIM_HandleTypeDef htim;

void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_TIM2_Init(void);

int main(void)
{
    HAL_Init();
    SystemClock_Config();
    MX_GPIO_Init();
    MX_TIM2_Init();

    while (1)
    {
        // 逐渐增加占空比,使 LED 亮度逐渐增加
        for (int i = 0; i <= 100; i++)
        {
            setPWM_DutyCycle(&htim, TIM_CHANNEL_1, i);
            HAL_Delay(10);
        }

        // 逐渐减小占空比,使 LED 亮度逐渐减小
        for (int i = 100; i >= 0; i--)
        {
            setPWM_DutyCycle(&htim, TIM_CHANNEL_1, i);
            HAL_Delay(10);
        }
    }
}

void SystemClock_Config(void)
{
    RCC_OscInitTypeDef RCC_OscInitStruct = {0};
    RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};

    /** Configure the main internal regulator output voltage
    */
    __HAL_RCC_PWR_CLK_ENABLE();
    __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1);

    /** Initializes the RCC Oscillators according to the specified parameters
    * in the RCC_OscInitTypeDef structure.
    */
    RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
    RCC_OscInitStruct.HSEState = RCC_HSE_ON;
    RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
    RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
    RCC_OscInitStruct.PLL.PLLM = 4;
    RCC_OscInitStruct.PLL.PLLN = 180;
    RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2;
    RCC_OscInitStruct.PLL.PLLQ = 2;
    RCC_OscInitStruct.PLL.PLLR = 2;
    if (HAL_RCC_OscConfig(&RCC_OscInitStruct)!= HAL_OK)
    {
        Error_Handler();
    }

    /** Activate the Over-Drive mode
    */
    if (HAL_PWREx_EnableOverDrive()!= HAL_OK)
    {
        Error_Handler();
    }

    /** Initializes the CPU, AHB and APB buses clocks
    */
    RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2;
    RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
    RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
    RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV4;
    RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2;

    if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_5)!= HAL_OK)
    {
        Error_Handler();
    }
}

static void MX_GPIO_Init(void)
{
    GPIO_InitTypeDef GPIO_InitStruct = {0};

    /* GPIO Ports Clock Enable */
    __HAL_RCC_GPIOD_CLK_ENABLE();

    /*Configure GPIO pin Output Level */
    HAL_GPIO_WritePin(GPIOD, GPIO_PIN_12 | GPIO_PIN_13 | GPIO_PIN_14 | GPIO_PIN_15, GPIO_PIN_RESET);

    /*Configure GPIO pins : PD12 PD13 PD14 PD15 */
    GPIO_InitStruct.Pin = GPIO_PIN_12 | GPIO_PIN_13 | GPIO_PIN_14 | GPIO_PIN_15;
    GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
    HAL_GPIO_Init(GPIOD, &GPIO_InitStruct);
}

static void MX_TIM2_Init(void)
{
    TIM_ClockConfigTypeDef sClockSourceConfig = {0};
    TIM_MasterConfigTypeDef sMasterConfig = {0};
    TIM_OC_InitTypeDef sConfigOC = {0};

    htim.Instance = TIM2;
    htim.Init.Prescaler = 100 - 1;
    htim.Init.CounterMode = TIM_COUNTERMODE_UP;
    htim.Init.Period = 100 - 1;
    htim.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
    if (HAL_TIM_Base_Init(&htim)!= HAL_OK)
    {
        Error_Handler();
    }
    sClockSourceConfig.ClockSource = TIM_CLOCKSOURCE_INTERNAL;
    if (HAL_TIM_ConfigClockSource(&htim, &sClockSourceConfig)!= HAL_OK)
    {
        Error_Handler();
    }
    if (HAL_TIM_PWM_Init(&htim)!= HAL_OK)
    {
        Error_Handler();
    }
    sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET;
    sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE;
    if (HAL_TIMEx_MasterConfigSynchronization(&htim, &sMasterConfig)!= HAL_OK)
    {
        Error_Handler();
    }
    sConfigOC.OCMode = TIM_OCMODE_PWM1;
    sConfigOC.Pulse = 0;
    sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH;
    sConfigOC.OCFastMode = TIM_OCFAST_DISABLE;
    if (HAL_TIM_PWM_ConfigChannel(&htim, &sConfigOC, TIM_CHANNEL_1)!= HAL_OK)
    {
        Error_Handler();
    }

    HAL_TIM_PWM_Start(&htim, TIM_CHANNEL_1);
}

void Error_Handler(void)
{
    __disable_irq();
    while (1)
    {
    }
}

void setPWM_DutyCycle(TIM_HandleTypeDef *htim, uint32_t channel, float dutyCycle)
{
    uint32_t period = htim->Instance->ARR;
    uint32_t pulse = (uint32_t)(dutyCycle * period / 100);
    __HAL_TIM_SET_COMPARE(htim, channel, pulse);
}

在这个示例中,我们使用 TIM2 的通道 1 产生 PWM 信号,控制连接在 PD12 引脚上的 LED 的亮度。通过逐渐改变 PWM 的占空比,实现了 LED 亮度的渐变效果。 

五、总结

        本文详细介绍了如何在 STM32F429 上使用 HAL 库实现 PWM 输出。通过配置定时器和通道,我们可以产生不同频率和占空比的 PWM 信号,用于控制各种外设。在实际应用中,可以根据具体需求调整参数,以满足不同的应用场景。希望本文对大家在使用 STM32F429 的 PWM 功能时有所帮助。

 

 

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

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

相关文章

Python的图像算术与逻辑运算详解

一.图像加法运算 图像加法运算主要有两种方法。第一种是调用Numpy库实现&#xff0c;目标图像像素为两张图像的像素之和&#xff1b;第二种是通过OpenCV调用add()函数实现。第二种方法的函数原型如下&#xff1a; dst add(src1, src2[, dst[, mask[, dtype]]]) – src1表示第…

JavaScript使用地理位置 API

前言 在JavaScript中&#xff0c;Geolocation API 是一种用于访问用户地理位置的接口。这个API允许网页应用程序获取用户的位置并提供基于位置的服务。 if (navigator.geolocation)navigator.geolocation.getCurrentPosition(function () {},function () {});这个函数中需要传…

VMware Workstation v17.6 中文注册精简版

VMware Workstation Pro 是一款功能强大的桌面虚拟化软件&#xff0c;它允许用户在单一的物理机器上同时运行多个操作系统。无论是在 Windows 或 Linux 主机上&#xff0c;您都可以轻松创建和运行各种虚拟机&#xff0c;包括 Windows、Linux 以及 BSD。 全新升级&#xff0c;更…

【debug】nvidia-smi:Failed to initialize NVML: Unknown Error

今天用服务器时又突然报错cuda不可用&#xff0c;输入nvidia-smi检查&#xff0c;报错如题。 根据网上搜的原因&#xff0c;可能是因为系统内核自动升级导致与驱动版本不匹配&#xff0c;尝试重启&#xff1a;reboot&#xff0c;报错&#xff1a; 尝试 exit 退出容器再进入&am…

android framework工程师遇到成长瓶颈迷茫怎么办?千里马经验分享

背景 近来有一些framework老司机粉丝朋友发来了一些framework工作中的一些疑问&#xff0c;具体描述如下&#xff1a; 这个同学遇到的问题&#xff0c;其实就是大部分framework开发者工作比较久后遇到的一个上升瓶颈问题。 具体总结有以下几个瓶颈问题 1、framework属于系…

Qt qrc机制

文章目录 0. 前言1. qrc机制2. qrc使用 0. 前言 要设置窗口图标&#xff0c;就需要有图片及其图片所在路径&#xff0c;在本机上可能没什么问题&#xff0c;但是换了一个机器&#xff0c;路径可能不一致或者图片丢失&#xff0c;这就导致图片显示不出来。 Qt引入qrc机制&…

基于springboot+vue实现的农家乐管理系统

基于springbootvue实现的山庄农家乐管理系统前后端分离项目&#xff08;文末查看源码lw&#xff09;4-10 系统角色&#xff1a; 管理员、用户 主要功能&#xff1a; &#xff08;1&#xff09;用户关键功能包含用户注册登陆、个人信息修改、首页、农家乐、美食信息、民宿信息…

Autosar(Davinci) --- 创建一个S/R类型的port(下)

前言: 前面章节我们讲解了S/R类型的Port如何创建,这一章节,我们着重讲一下生成的代码,以及我们如何添加代码让这些门与灯之间的关系产生连接。 一、CtSaDoor.c 在【Rte.c】的【IO_TASK】中我们可以看到,反复的判断Rte_Ev_Cyclic_IO_Task_0_200ms这个条件是否成立,当200…

FreeRTOS学习笔记(四)Freertos的中断管理及临界保护

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、Cortex-M 中断管理1.1 中断优先级分组1.2 相关寄存器1.3 相关宏定义1.4 FreeRTOS 开关中断 二、临界段及其保护2.1 taskENTER_CRITICAL( ) 和 taskEXIT_CRI…

【VUE】跨域问题的概念,以及解决方法。

目录 1.跨域概念 2.解决方法 2.1 配置网络请求代理 2.2 使用CrossOrigin 注解 2.3 通过配置文件实现跨域 2.4 添加 CorsWebFilter 来解决跨域问题 1.跨域概念 跨域问题是由于浏览器实施了同源策略&#xff0c;该策略要求请求的域名、协议和端口必须与提供资源的服务相同。…

828华为云征文|部署RedisStack+可视化操作

目录 1.介绍 1.1 什么是华为云Flexus X实例 1.2 什么是Redis Stack ​编辑 2 准备华为云Flexus应用服务器L实例 3 准备docker环境 4 docker 安装 Redis Stack 4.1拉取镜像 4.2 确认镜像 4.3 启动容器 4.4 放行安全组端口 4.5 操作redis 5 docker compose 配置持久 1.…

Vision Transformer(ViT)模型原理及PyTorch逐行实现

Vision Transformer(ViT)模型原理及PyTorch逐行实现 一、TRM模型结构 1.Encoder Position Embedding 注入位置信息Multi-head Self-attention 对各个位置的embedding融合&#xff08;空间融合&#xff09;LayerNorm & ResidualFeedforward Neural Network 对每个位置上单…

LabVIEW FIFO详解

在LabVIEW的FPGA开发中&#xff0c;FIFO&#xff08;先入先出队列&#xff09;是常用的数据传输机制。通过配置FIFO的属性&#xff0c;工程师可以在FPGA和主机之间&#xff0c;或不同FPGA VIs之间进行高效的数据传输。根据具体需求&#xff0c;FIFO有多种类型与实现方式&#x…

SpringSecurity原理解析(二):认证流程

1、SpringSecurity认证流程包含哪几个子流程&#xff1f; 1&#xff09;账号验证 2&#xff09;密码验证 3&#xff09;记住我—>Cookie记录 4&#xff09;登录成功—>页面跳转 2、UsernamePasswordAuthenticationFilter 在SpringSecurity中处理认证逻辑是在UsernamePas…

iOS——线程安全、线程同步与线程通信

线程安全和线程同步 线程安全&#xff1a;如果你的代码所在的进程中有多个线程在同时运行&#xff0c;而这些线程可能会同时运行这段代码。如果每次运行结果和单线程运行的结果是一样的&#xff0c;而且其他的变量的值也和预期的是一样的&#xff0c;就是线程安全的。 若每个…

18055 主对角线上的元素之和

### 思路 1. 输入一个3行4列的整数矩阵。 2. 计算主对角线上的元素之和。 3. 输出主对角线上的元素之和。 ### 伪代码 1. 初始化一个3行4列的矩阵 matrix。 2. 输入矩阵的元素。 3. 初始化一个变量 sum 为0&#xff0c;用于存储主对角线元素之和。 4. 遍历矩阵的行&#xff0c…

【Day08-IO-文件字节流】

File 1. 概述 File对象既可以代表文件、也可以代表文件夹。它封装的对象仅仅是一个路径名&#xff0c;这个路径可以存在&#xff0c;也可以不存在 构造器 说明 public File​(String pathname) 根据文件路径创建文件对象 public File​(String parent, String child) 根据…

vscode中使用go环境配置细节

1、在docker容器中下载了go的sdk 2、在/etc/profile.d/go.sh里填入如下内容: #!/bin/bashexport GOROOT=/home/ud_dev/go export PATH=$GOROOT/bin:$PATH 3、设置go env go env -w GOPROXY=https://goproxy.cn,direct go env -w GO111MODULE=on 4、重启这个容器,使得vscod…

DBAPI如何使用内存缓存

背景 在使用DBAPI创建API的时候&#xff0c;有时候SQL查询比较耗时&#xff0c;如果业务上对数据时效性要求不高&#xff0c;这种耗时的SQL可以使用缓存插件来将数据缓存起来&#xff0c;避免重复查询。 一般来说&#xff0c;可以使用redis memcache等缓存服务来存储缓存数据。…

活动|华院计算宣晓华受邀出席“AI引领新工业革命”大会,探讨全球科技的最新趋势

8月31日&#xff0c;“AI引领新工业革命”大会于上海图书馆圆满落幕。本次大会由TAA校联会和台协科创工委会联合主办&#xff0c;得到上海市台办、上海市台联、康师傅的大力支持。大会邀请了NVIDIA全球副总裁、亚太区企业营销负责人刘念宁&#xff0c;元禾厚望资本创始合伙人潘…