STM32应用开发——使用PWM+DMA驱动WS2812

STM32应用开发——使用PWM+DMA驱动WS2812

目录

  • STM32应用开发——使用PWM+DMA驱动WS2812
    • 前言
    • 1 硬件介绍
      • 1.1 WS2812介绍
        • 1.1.1 芯片简介
        • 1.1.2 引脚描述
        • 1.1.3 工作原理
        • 1.1.4 时序
        • 1.1.5 传输协议
      • 1.2 电路设计
    • 2 软件编程
      • 2.1 软件原理
      • 2.2 测试代码
        • 2.2.1 底层驱动
        • 2.2.2 灯效应用
      • 2.3 运行测试
        • 2.3.1 时序测试
        • 2.3.2 实际效果
    • 结束语

前言

串行灯带的应用十分广泛,其中以WS2812最为经典,这种灯带一般都是通过单总线的方式来驱动,也就是由一根数据线按照特定的时序输出,继而驱动灯带。这种方式在硬件和软件上都非常简单,但是如果软件用GPIO模拟时序的话比较占用主线程的资源,因此,如果能用硬件外设(比如PWM、SPI、串口)来模拟出这个时序,就能节省MCU的资源。
本文以PWM+DMA为例介绍如何驱动WS2812。

1 硬件介绍

1.1 WS2812介绍

1.1.1 芯片简介

WS2812是一款智能控制LED光源,其外观采用最新的MOLDING封装技术、控制电路和RGB芯片集成在2020组件的封装中。其内部包括智能数字端口数据锁存和信号整形放大驱动电路。还包括精密内部振荡器和电压可编程恒流控制部分,有效保证像素点光源的颜色。

1.1.2 引脚描述
引脚名称描述
DO数据输出控制数据输出到下一个芯片
GND电源负极
DI数据输入控制数据输入
VDD电源电源正极
1.1.3 工作原理

通过级联法把每个灯的DI和DO引脚首尾相连,数据可以从第一个IC开始,不断的传输到后面每一个IC,从而实现整条灯带的控制。
在这里插入图片描述

1.1.4 时序

WS2812通过不同的时序来表示0码1码复位码,如下图所示:
在这里插入图片描述
其中各信号的电平如下图所示:
在这里插入图片描述
注:不同型号的芯片在时序上会有点差异,具体以芯片数据手册为准。

1.1.5 传输协议

传输过程如下图所示:
在这里插入图片描述

每一个灯珠的RGB数据排列如下:
在这里插入图片描述

1.2 电路设计

WS2812的控制方法很简单,每个灯珠首尾相接进行级联即可,如下图所示:
在这里插入图片描述
其中,第一个灯珠的DI引脚接入到MCU的一个GPIO上面。

我这里使用STM32F103来作为主控MCU,引脚接线如下:

MCU引脚灯带引脚描述
PA0DI由MCU发送控制信号输入到灯带

2 软件编程

2.1 软件原理

通过DMA可以精确控制PWM输出的每一个方波,然后通过调整占空比,就可以输出0码1码复位码,从而实现灯珠的驱动。
举个例子:按照上面的手册的时序要求,每一个逻辑电平周期在1.25us左右,也就是800kHz,那么PWM输出的频率就可以设置为800kHz。此时改变PWM的占空比,就可以区分编码“0”和编码“1”,比如编码“0”的高电平脉宽和低电平脉宽分别为0.4us和0.85us,那么对应的PWM占空比就是32%和68%,然后通过DMNA连续传输RGB数据就可以实现灯带的颜色和亮度控制。

测试电平时序如下:

逻辑电平脉宽PWM占空比
逻辑0高电平0.40±0.15us32%
逻辑0低电平0.85±0.15us68%
逻辑1高电平0.85±0.15us68%
逻辑1低电平0.40±0.15us32%
复位低电平1.25±0.60us0%

2.2 测试代码

根据上述原理,编写测试代码。

2.2.1 底层驱动

ws2812_driver.h :

#ifndef __WS2812_DRIVER_H
#define __WS2812_DRIVER_H#include "stm32f10x.h"
#include "stm32f10x_conf.h"#define TIM2_CCR1_Address 0x40000034  // stm32 tim2 base address offset 0x34#define LED_NUM     8    // LED的数量
#define RGB_BIT     24   // 每个灯有24bit的RGB数据,依次按G R B排列#define RESET_WORD  5    // 在传输RGB数据前保持一段低电平
#define DUMMY_WORD  5    // 在传输RGB数据后保持一段低电平#define TIMING_0    29   // T0H(32%) = 1.25us * (29 / 90) = 0.40us, T0L(68%) = 1.25 - 0.40 = 0.85us 
#define TIMING_1    61   // T1H(68%) = 1.25us * (61 / 90) = 0.85us, T1L(32%) = 1.25 - 0.85 = 0.40us void led_display(uint8_t (*led_buf)[3], uint8_t led_num);
void ws2812_init(void);#endif

ws2812_driver.c :

#include "ws2812_driver.h"
#include "string.h"uint16_t pwm_dma_buf[RESET_WORD + RGB_BIT * LED_NUM + DUMMY_WORD];void pwm_init(void)
{GPIO_InitTypeDef GPIO_InitStructure;TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;TIM_OCInitTypeDef  TIM_OCInitStructure;RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO, ENABLE);GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;			GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA, &GPIO_InitStructure);GPIO_ResetBits(GPIOA, GPIO_Pin_0);RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);TIM_TimeBaseStructure.TIM_Period = 90 - 1;     // 72MHz / 90 = 800kHzTIM_TimeBaseStructure.TIM_Prescaler = 0;TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);/* PWM2 Mode configuration: Channel1 */TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1;TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;TIM_OCInitStructure.TIM_Pulse = 50;TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High;TIM_OCInitStructure.TIM_OCIdleState = TIM_OCIdleState_Reset;TIM_OC1Init(TIM2, &TIM_OCInitStructure);TIM_OC1PreloadConfig(TIM2, TIM_OCPreload_Enable);// TIM_ARRPreloadConfig(TIM2, ENABLE);/* TIM2 enable counter */TIM_Cmd(TIM2, ENABLE);
}void pwm_dma_init(void)
{/* configure DMA */DMA_InitTypeDef DMA_InitStructure;//定义DMA初始化结构体/* DMA clock enable */RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);	//使能DMA时钟(用于SPI的数据传输)memset((uint8_t*)&pwm_dma_buf, 0, sizeof(pwm_dma_buf));/* DMA1 Channel5 Config for PWM2 by TIM2_CH1*/DMA_DeInit(DMA1_Channel5);DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)TIM2_CCR1_Address;	// physical address of Timer 3 CCR1DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)&pwm_dma_buf;		// this is the buffer memory DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;						// data shifted from memory to peripheralDMA_InitStructure.DMA_BufferSize = sizeof(pwm_dma_buf)/2;DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;					// automatically increase buffer indexDMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;							// stop DMA feed after buffer size is reachedDMA_InitStructure.DMA_Priority = DMA_Priority_Medium;DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;DMA_Init(DMA1_Channel5, &DMA_InitStructure);/* TIM2 DMA Request enable */TIM_DMACmd(TIM2, TIM_DMA_CC1, ENABLE);TIM_DMACmd(TIM2, TIM_DMA_Update, ENABLE);
}void pwm_dma_send(void)
{DMA_SetCurrDataCounter(DMA1_Channel5, sizeof(pwm_dma_buf)/2); 	// load number of bytes to be transferredDMA_Cmd(DMA1_Channel5, ENABLE); 			// enable DMA channel 5TIM_Cmd(TIM2, ENABLE); 						// enable Timer 2while(!DMA_GetFlagStatus(DMA1_FLAG_TC5)) ; 	// wait until transfer completeDMA_Cmd(DMA1_Channel5, DISABLE); 			// disable DMA channel 5DMA_ClearFlag(DMA1_FLAG_TC5); 				// clear DMA1 Channel 5 transfer complete flagTIM_Cmd(TIM2, DISABLE); 	// disable Timer 2
}void led_display(uint8_t (*led_buf)[3], uint8_t led_num)
{uint8_t i, j;// led_buf -> pwm_dma_buffor(i = 0; i < led_num; i++){// N ledfor(j = 0; j < 8; j++){// 1 color -> 8bit// gpwm_dma_buf[RESET_WORD+RGB_BIT*i+j] = ((led_buf[i][1] << j) & 0x80) ? TIMING_1 : TIMING_0;// rpwm_dma_buf[RESET_WORD+RGB_BIT*i+j+8] = ((led_buf[i][0] << j) & 0x80) ? TIMING_1 : TIMING_0;// bpwm_dma_buf[RESET_WORD+RGB_BIT*i+j+16] = ((led_buf[i][2] << j) & 0x80) ? TIMING_1 : TIMING_0;}}// pwm startpwm_dma_send();
}void ws2812_init(void)
{pwm_init();pwm_dma_init();
}
2.2.2 灯效应用

ws2812_app.h :

#ifndef __WS2812_APP_H
#define __WS2812_APP_H#include "stm32f10x.h"
#include "stm32f10x_conf.h"
#include "ws2812_driver.h"typedef enum 
{LED_MODE_OFF,LED_MODE_ALL_ON,	LED_MODE_BREATHE,	LED_MODE_GRADIENT,LED_MODE_FLOW,	
}led_mode_t;typedef struct
{led_mode_t mode;  uint8_t g;                uint8_t r;              uint8_t b;              uint8_t brightness;  
}led_t;void led_init(void);
void led_handle(void);#endif

ws2812_app.c :

#include "ws2812_app.h"led_t led;
uint8_t rgb_buf[LED_NUM][3];void led_init(void)
{ws2812_init();led.mode = LED_MODE_ALL_ON;led.r = 0x00;led.g = 0xE0;led.b = 0x80;
}void led_handle(void)
{uint8_t i;switch (led.mode){case LED_MODE_OFF:for (i = 0; i < LED_NUM; i++){rgb_buf[i][0] = 0;  // rrgb_buf[i][1] = 0;  // grgb_buf[i][2] = 0;  // b}break;case LED_MODE_ALL_ON:for (i = 0; i < LED_NUM; i++){rgb_buf[i][0] = led.r;  // rrgb_buf[i][1] = led.g;  // grgb_buf[i][2] = led.b;  // b}break;// ......可以自己加入更多的灯效default:break;}led_display(rgb_buf, LED_NUM);
}

main.c :

#include "sys.h"
#include "delay.h"
#include "usart.h"
#include "ws2812_app.h"int main(void)
{SystemInit();delay_init();led_init();while(1){led_handle();delay_ms(5);}
}

2.3 运行测试

2.3.1 时序测试

使用逻辑分析仪抓取信号,得到的结果如下:

  1. 8个LED连续写入RGB值:
    在这里插入图片描述

  2. 编码1高电平(T1H)850ns:
    在这里插入图片描述

  3. 编码1低电平(T1L)400ns:
    在这里插入图片描述

  4. 编码1周期1.25us:
    在这里插入图片描述

  5. 编码0高电平(T0H)400ns:
    在这里插入图片描述

  6. 编码0高电平(T0H)850ns:
    在这里插入图片描述

  7. 编码0周期1.25us:
    在这里插入图片描述

结论:实际输出的波形和理论一致。

2.3.2 实际效果

用在线颜色选取器把代码设置的颜色值输入进去,得到该颜色,然后和实际灯带点亮的颜色比对。

  1. 颜色拾取器显示如下:
    在这里插入图片描述
  2. 实际灯带颜色如下:
    在这里插入图片描述

结论:灯带实际显示的颜色准确无误。

结束语

关于stm32如何使用PWM+DMA驱动WS2812的讲解就到这里,如果还有什么问题,欢迎在评论区留言。

源码下载链接

如果这篇文章能够帮到你,就…你懂的。
在这里插入图片描述

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

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

相关文章

leetcode二叉树相关题目

目录 二叉树的建立整数数组转二叉树Object数组转二叉树 二叉树的遍历leetcode94.二叉树的中序遍历leetcode144.二叉树的前序遍历 二叉树的建立 整数数组转二叉树 下面只是一个简单的示例&#xff0c;没考虑某个子树为空的情况。把{1, 2, 3, 21, 22, 31, 32} 转变为一个二叉树…

新兴AI技术及其创业机会

量子计算与AI 量子计算是未来计算技术的前沿&#xff0c;它通过量子比特进行信息处理&#xff0c;相较于传统计算机&#xff0c;量子计算在处理复杂问题上有着天然的优势。将量子计算与AI结合&#xff0c;可以极大提升AI模型训练的效率和处理数据的能力。 创业机会&#xff1a…

基于NXP的ISELED开发环境搭建

目录 前言一、获取S32K SDK ISELED driver二、安装S32K SDK ISELED driverISELED文件说明三、ISELED工程介绍方案框图:图形化界面配置注意事项前言 NXP S32K ISELED 32位Arm Cortex-M微控制器 (MCU) 包括一种全新的通信协议 (ISELED),用于控制汽车和工业照明应用中的静态和动…

简述前端开发全景:从基础到未来的语言与趋势解析

前端开发概述 前端开发指的是创建网站或应用程序中用户可以直接看到和与之交互的部分。前端开发的主要目标是通过编写代码来制作用户界面&#xff0c;确保信息以易于理解和互动的方式展示给用户。 HTML和CSS HTML&#xff1a;HTML是构建网页的基础&#xff0c;负责定义网页的…

系统IO函数接口

目录 前言 一. man手册 1.1 man手册如何查询 1.2 man手册基础 二.系统IO函数接口 三.open打开文件夹 3.1 例1 open打开文件 3.2 open打开文件代码 3.3 例2 创建文件 四.write写文件 4.1 write写文件 五. read读文件 5.1 read读文件与偏移 5.2 偏移细节 5.3 read读文件代码 六.复…

3.5网安学习第三阶段第五周回顾(个人学习记录使用)

本周重点 ①SSRF服务器端请求伪造 ②序列化和反序列化 ③Vaudit代码审计 本周主要内容 ①SSRF服务器端请求伪造 一、概述 SSRF: server site request forgery (服务器端请求伪造)。 SSR: 服务端请求&#xff0c;A服务器通过函数向B服务器发送请求。 SSRF发生的前提条件…

Flutter动画笔记---总结

三种情况&#xff1a;第一种情况&#xff1a;隐式动画&#xff0c;全自动动画&#xff08;属于内部封装好了&#xff0c;只需要几行代码&#xff0c;就可以实现非常强大的效果&#xff09;第二种情况&#xff1a;显示动画&#xff0c;手动控制的动画&#xff08;提供自定义选择…

【ARM 嵌入式 C 字符串系列 23 -- 返回从父字符串找到相同子字符串的个数】

请阅读【嵌入式开发学习必备专栏 】 文章目录 返回从父字符串找到相同子字符串的个数 返回从父字符串找到相同子字符串的个数 在 C 语言中&#xff0c;可以编写一个自定义函数来计算一个父字符串&#xff08;haystack&#xff09;中相同子字符串&#xff08;needle&#xff09…

蓝桥杯备考随手记: practise04

问题描述: 小明被不明势力劫持。后莫名其妙被扔到 X 星站再无问津。小明得知每天都有飞船飞往地球&#xff0c;但需要 1 元的船票&#xff0c;而他却身无分文。他决定在 X 星站打工。好心的老板答应包食宿&#xff0c;第 1 天给他 1 元钱。并且&#xff0c;以后的每一天都比前…

《QT实用小工具·三》偏3D风格的异型窗体

1、概述 源码放在文章末尾 可以在窗体中点击鼠标左键进行图片切换&#xff0c;项目提供了一些图片素材&#xff0c;整体风格偏向于3D类型&#xff0c;也可以根据需求自己放置不同的图片。 下面是demo演示&#xff1a; 项目部分代码如下所示&#xff1a; 头文件部分&#xff…

Linux安装Tomcat保姆级教程

文章目录 前言一、安装JDK二、Tomcat下载三、Tomcat安装1.创建Tomcat的安装目录2.切换到Tomcat的安装目录3.上传Tomcat安装包4.解压缩5.切换到安装好的tomcat的bin目录6.启动tomcat7.网络请求测试 四、外部电脑访问nginx设置方法一&#xff1a;关闭防火墙方法二&#xff1a;添加…

Java反射系列(3):从spring反射工具ReflectionUtils说起

目录 传送门 兼容性引发的"血案" ReflectionUtils的原理 目的有三 ReflectionUtils的API使用 Method getAllDeclaredMethods findMethod invokeMethod Field getDeclaredFields findField getField makeAccessible Constructor accessibleConstructo…

Spring IoCDI(2)

IoC详解 通过上面的案例, 我们已经知道了IoC和DI的基本操作, 接下来我们来系统地学习Spring IoC和DI的操作. 前面我们提到的IoC控制反转, 就是将对象的控制权交给Spring的IoC容器, 由IoC容器创建及管理对象. (也就是Bean的存储). Bean的存储 我们之前只讲到了Component注解…

SAP Fiori开发中的JavaScript基础知识10 - 变量作用域,变量提升

1. 背景 本文将会介绍JavaScript中两个相对复杂的概念&#xff0c;也即作用域和变量提升。这个概念在JavaScript中的设计和ABAP中的设计区别还是比较大的&#xff0c;特别是变量提升部分。第一次接触JavaScript的同学&#xff0c;要注意理解和区分。 2. 作用域 在JavaScript…

vscode安装

&#x1f308;个人主页&#xff1a;Rookie Maker &#x1f3c6;&#x1f3c6;关注博主&#xff0c;随时获取更多关于IT的优质内容&#xff01;&#x1f3c6;&#x1f3c6; &#x1f600;欢迎来到小田代码世界~ &#x1f601; 喜欢的小伙伴记得一键三连哦 ૮(˶ᵔ ᵕ ᵔ˶)ა …

商场促销--策略模式

1.1 商场收银软件 package com.lhx.design.pattern.test;import java.util.Scanner;public class Test {public static void main(String[] args){System.out.println("**********************************************"); System.out.println("《大话设计模式…

vue3鼠标向下滑动,导航条改变背景颜色和logo的封装

代码中使用了element-plus组件&#xff0c;需先安装 向下滑动前 向下滑动后&#xff08;改变了logo 字体 背景颜色&#xff09; <script lang"ts" setup> import router from /router; import { ArrowDown } from element-plus/icons-vue import { ref, …

【tensorflow框架神经网络实现鸢尾花分类—优化器】

文章目录 1、前言2、神经网络参数优化器2.1、SGD2.2、SGDM2.3、Adagrad2.4、RMSProp2.5、Adam 3、实验对比不同优化器4、结果对比 1、前言 此前&#xff0c;在【tensorflow框架神经网络实现鸢尾花分类】一文中使用梯度下降算法SGD&#xff0c;对权重 w w w和偏置 b b b进行更新…

如何在Windows上安装SSH

SSH&#xff08;Secure Shell&#xff09;协议是一种强大且安全的网络协议&#xff0c;它为用户提供了一种加密的方式来远程访问和管理终端设备。在Windows操作系统中&#xff0c;安装和配置SSH服务是一个相对直接的过程&#xff0c;可以让你享受到SSH带来的便利和安全性。下面…

软件设计原则:迪米特法则

定义 迪米特法则&#xff08;Law of Demeter, LoD&#xff09;&#xff0c;又称最少知识原则&#xff0c;它指导我们在设计软件时&#xff0c;应当尽量减少对象之间的交互&#xff0c;一个对象应该对其他对象有尽可能少的了解。具体来说&#xff0c;一个对象应该只调用属于以下…