关于STM32利用TIM+PWM+DMA控制WS2812

开发环境

MCU:STM32F103c8t6
开发工具:STM32CubeMX

 使用板子参考原理图:STM32F103C8T6最小系统板开源链接

PWM

脉宽调制(PWM)基本原理:控制方式就是对逆变电路开关器件的通断进行控制,使输出端得到一系列幅值相等但宽度不一致的脉冲,用这些脉冲来代替正弦波或所需要的波形。也就是在输出波形的半个周期中产生多个脉冲,使各脉冲的等值电压为正弦波形,所获得的输出平滑且低次谐波少。按一定的规则对各脉冲的宽度进行调制,既可改变逆变电路输出电压的大小,也可改变输出频率 。

通过数据手册我们可以看到TIM2,TIM3,TIM4(通用定时器)挂载在APB2总线上,每个通用定时器都有独立的4个通道可以用来作为:输入捕获、输出比较、PWM输出、单脉冲模式输出等。

那么直接上STM32CUBEMX

 这边选用了PB9作为TIM4_CH4通道用来做PWM输出(上述勾选去掉)

 也就是外部时钟TCLK=72mhz

同时引进定时器的原理 向上计数模式UP

 ARR就是自动重装载值

 CCRX为捕获/比较寄存器值

 CNT为计数器当前值

那么其中的逻辑是这样的

  • 当CNT小于CCRx时,TIMx_CHx通道输出设置的电平;
  • 当CNT等于或大于CCRx时,TIMx_CHx通道输出与设置相反的电平。

 设置的电平

那么我们直接配置好

中加入使能代码

HAL_TIM_PWM_Start(&htim4,TIM_CHANNEL_4);

看下示波器效果

看的出来是低电平20% 高电平80% 频率2KHZ

那么是怎么配置出来的呢

看图可得出以下数据:

PSC=71

向上计数模式

ARR=499

>>>>>>

PWM 模式1

CCRX=100

>>>>>>

通道输出极性低电平

好的,我们得到这么多数据,还需要记住一些公式

Fpwm=TCLK/(ARR+1)*(PSC+1) =2000HZ

占空比=CCRX/(ARR+1)=100/(499+1)=20%

改CCR1可以修改占空比,修改arr可以修改频率

好的,这样简单的PWM配置就完成了,接下来模拟时序

WS2812时序讲解具体在这个博客,看完再跳回本页面

1码 2/3高电平 1/3低电平

0码 1/3高电平 2/3低电平

一个码的周期是1.25us,也就是800khz

无聊写了个代码,算对应的PSC和ARR

计算 定时器arr和psc

随便选一组,怎么方便怎么来

选PSC=1,ARR=44,ch polarity high.

波形符合理论

 引入新的方式:直接修改CCRx寄存器的值

htim4.Instance->CCR4 = 30

DMA

进入正题 TIM+DMA配置

 

(HAL_TIM_PWM_PulseFinishedCallback() 是一个回调函数,当DMA传输完成以后,就会调用这个函数,由于本文DMA传输模式选择为Circular,所以DMA需要手动关闭,否则DMA会不断的搬运数据。)

DMA传输位宽和定义的缓冲区位宽要一致

u32 对word

u16 对half word

u8  对 byte

然后生成文件,打开工程

新建一个 NEW GROUP

创建WS2812.h   WS2812.c文件 

#ifndef _WS2812_H
#define _WS2812_H
#endif
//标识的命名规则一般是头文件名全大写,前后加下划线,并把文件名中的“.”也变成下划线
//头文件区
#include "main.h"
#include "dma.h"
#include "tim.h"


//用户修改参数区
#define ONE_PULSE        (59)                           //1 码计数个数
#define ZERO_PULSE       (29)                           //0 码计数个数
#define RESET_PULSE      (48)                           //80 复位电平个数(不能低于40)
#define LED_NUMS         (4)                            //led 个数
#define LED_DATA_LEN     (24)                           //led 长度,单个需要24个字节
#define WS2812_DATA_LEN  (LED_NUMS*LED_DATA_LEN)        //ws2812灯条需要的数组长度



void ws2812_set_RGB(uint8_t R, uint8_t G, uint8_t B, uint16_t num);//设置彩灯颜色
void ws2812_example(void);
void HAL_TIM_PWM_PulseFinishedCallback(TIM_HandleTypeDef *htim);


uint16_t  RGB_buffur[RESET_PULSE + WS2812_DATA_LEN] = { 0 };

WS2812_DATA_LEN ==(LED_NUMS*LED_DATA_LEN)

测试用的LED_NUMS=4,那么数组长度为4*24+reset_pulse       

reset_pulse>=40

(DMA 每一个定时器周期就搬运一个0到定时器CCR中,定时器将产生一个1.25us的全低电平,40个为50us,这个50us的低电平作为ws2812的复位信号。)

#include "WS2812.h"


 uint16_t  RGB_buffur[RESET_PULSE + WS2812_DATA_LEN] = { 0 };


void ws2812_set_RGB(uint8_t R, uint8_t G, uint8_t B, uint16_t num)
{
    //指针偏移:需要跳过复位信号的N个0
    uint16_t* p = (RGB_buffur + RESET_PULSE) + (num * LED_DATA_LEN);
    
    for (uint16_t i = 0;i < 8;i++)
    {
        //填充数组
        p[i]      = (G << i) & (0x80)?ONE_PULSE:ZERO_PULSE;
        p[i + 8]  = (R << i) & (0x80)?ONE_PULSE:ZERO_PULSE;
        p[i + 16] = (B << i) & (0x80)?ONE_PULSE:ZERO_PULSE;
    }

}

void HAL_TIM_PWM_PulseFinishedCallback(TIM_HandleTypeDef *htim)
{
//    HAL_TIM_PWM_Stop_DMA(&htim4,TIM_CHANNEL_4);
//    HAL_TIM_PWM_Stop_DMA(&htim1,TIM_CHANNEL_1);
	HAL_TIM_PWM_Stop_DMA(&htim4,TIM_CHANNEL_3);//PA8
}

void ws2812_example(void)
{
    //#1.填充数组
    ws2812_set_RGB(0x22, 0x00, 0x00, 0);
    ws2812_set_RGB(0x00, 0x22, 0x00, 1);
    ws2812_set_RGB(0x00, 0x00, 0x22, 2);
    ws2812_set_RGB(0x22, 0x22, 0x22, 3);
    //#2.传输数据
//    HAL_TIM_PWM_Start_DMA(&htim4,TIM_CHANNEL_4,(uint32_t *)RGB_buffur,(176));  
//	  HAL_TIM_PWM_Start_DMA(&htim1,TIM_CHANNEL_1,(uint32_t *)RGB_buffur,(176));
	HAL_TIM_PWM_Start_DMA(&htim4,TIM_CHANNEL_3,(uint32_t *)RGB_buffur,(176));
    //#3.延时:使效果可以被观察
    HAL_Delay(500);
    
    ws2812_set_RGB(0x22, 0x00, 0x00, 1);
    ws2812_set_RGB(0x00, 0x22, 0x00, 2);
    ws2812_set_RGB(0x00, 0x00, 0x22, 3);
    ws2812_set_RGB(0x22, 0x22, 0x22, 0);
    
//    HAL_TIM_PWM_Start_DMA(&htim4,TIM_CHANNEL_4,(uint32_t *)RGB_buffur,(176)); 
//    HAL_TIM_PWM_Start_DMA(&htim1,TIM_CHANNEL_1,(uint32_t *)RGB_buffur,(176));
    HAL_TIM_PWM_Start_DMA(&htim4,TIM_CHANNEL_3,(uint32_t *)RGB_buffur,(176));	
    HAL_Delay(500);
	}

(RGB_buffur + RESET_PULSE) + (num * LED_DATA_LEN)对应得是

RGB_buffur[RESET_PULSE + num * LED_DATA_LEN]的地址(num取值这里是0-3)

实际上就是跳过最开始的数组里面的RESET_PULSE

RGB_buffur[]={ RESET_PULSE ,NUM0数据,NUM1数据,NUM2数据,NUM3数据}

在main.c文件while中加入

		ws2812_example();

实物效果图

WS2812

参考链接

关于STM32F4xx使用DMA+TIM3_PWM调试灯带WS2812过程记录

STM32F4可以通过TIM PWM DMA控制WS2812灯带。 在使用STM32F4来控制WS2812灯带之前,我们需要了解一些基本的原理。WS2812灯带是一种基于Neopixel技术的RGB LED灯带,它具有灯珠之间串行通信的特点。通过发送一系列的0和1的数据信号,可以控制每个灯珠的颜色与亮度。 首先,我们需要配置STM32F4的GPIO引脚作为TIM输出模式,选择PWM模式,并配置DMA进行数据传输。接着,我们需要设置TIM的周期和预分频系数,以控制PWM信号的频率和占空比。根据WS2812的通信协议,每个数据位以50%占空比的PWM信号来表示,其中逻辑0和逻辑1的时间分别为400ns和800ns。 然后,我们通过DMA传输具有正确占空比的PWM数据到GPIO引脚,以控制WS2812灯带。我们可以使用定时器计数器的更新事件作为触发源,通过TIMDMA请求信号来触发DMA传输。 我们可以通过编写相应的代码来配置STM32F4的定时器和DMA。首先,我们需要定义一个数据缓冲区,将要传输的PWM数据写入缓冲区中。然后,我们配置DMA的传输长度、传输方向和传输模式。接着,我们配置定时器的PWM模式、周期和预分频系数。最后,我们启动定时器和DMA,并等待传输完成。 以上就是利用STM32F4的TIM PWM DMA控制WS2812灯带的简要介绍。通过正确配置定时器和DMA,我们可以实现高效、准确地控制WS2812灯带的颜色和亮度,从而实现丰富多彩的灯光效果。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

姜浩鑫

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值