STM32103 PWM+DMA驱动WS2812B 寄存器版

最近项目上用到WS2812B的LED光源,主控芯片为STM32F103,经过一翻折腾,顺利点亮光源,现分享与大家,仅供参考!

程序没有用ST库,直接操作寄存器完成,如有什么问题,欢迎大家留言交流。

WS2812B的控制原理我这不多说,大家可以搜下,我们直接开始代码的实现。


程序中我们定义两个一维数组,一个是用来存放LED灯RGB的值,存放顺序如下(这个数组可以定义成二维数码)

另外一个数组用来存放前面数组转换成定时器的PWM值,DMA就是直接读取这些数据发送给定时器。其中第一和最后一个数据固定为0,意义就是为了让发送数据开始前和结束后一直是低电平,在整个程序运行过程中,我的定时器一直是在工作的,没有关闭。

PWM输出引脚我用的是定时器3的通道4(TIM3_CH4),触发的DMA的通道3

程序代码如下:

#ifndef __WS281x_H__
#define __WS281x_H__
 
#define TIMER_PERIOD    (60)  //定时器的自动重载值,控制PWM波形的周期
#define WS281x_0        (19)  //0码对应的比较值
#define WS281x_1        (38)  //1码对应的比较值
 
#define LED_NUM         (4)   //LED灯的数量
 
#define LEN_LED_BIT     (LED_NUM * 24 + 2)//加2:第一个和最后一个数据为0,使其输出为低电平
 
 
void WS281xInit(void);
void WS281xSend(void);
void SetRGBData(uint16_t NumLed,uint8_t ValG,uint8_t ValR,uint8_t ValB);
 
#endif // __WS281x_H__
uint8_t  ValRGB[LED_NUM * 3];
uint16_t ValPwmBuff[LEN_LED_BIT];
 
void WS281xInit(void)
{
 
	//定时器初始化
	RCC->APB1ENR  |= RCC_APB1ENR_TIM3EN;
	TIM3->CR1      = (TIM_CR1_CEN + TIM_CR1_ARPE);
	TIM2->CR2      = 0;
	TIM3->SMCR     = 0;
	TIM3->CCMR1    = 0;
	TIM3->CCMR2    = (TIM_CCMR2_OC4M_2+TIM_CCMR2_OC4M_1+TIM_CCMR2_OC4PE);
	TIM3->CCER     = TIM_CCER_CC4E;
	TIM3->PSC      = 0;
	TIM3->ARR      = (TIMER_PERIOD-1);           //0码-19;1码-38
	TIM3->DIER     = (TIM_DIER_TDE+TIM_DIER_UDE);//允许更新触发DMA请求
 
	//DMA初始化
	RCC->AHBENR   |= RCC_AHBENR_DMA1EN;
	DMA1_Channel3->CCR  &= ~DMA_CCR3_EN;
	DMA1_Channel3->CNDTR = 0;//先设置为0,DMA触发了也不会发送数据
	DMA1_Channel3->CPAR  = (uint32_t)&TIM3->CCR4;
	DMA1_Channel3->CMAR  = (uint32_t)&ValPwmBuff[0];
	DMA1_Channel3->CCR   = 0x3591;
}
 
//设置对应LED的RGB值
void SetRGBData(uint16_t NumLed,uint8_t ValG,uint8_t ValR,uint8_t ValB)
{
	if ((NumLed == 0) || (NumLed > LED_NUM)) return;
	NumLed--;
	ValRGB[NumLed * 3 + 0] = ValG;
	ValRGB[NumLed * 3 + 1] = ValR;
	ValRGB[NumLed * 3 + 2] = ValB;
}
 
//调用此函数发送数据时,确保上次数据已经发送完成
void WS281xSend(void)
{
	uint8_t Data,i,j,*pLedDat;
	uint16_t *pLedPwmBuf;
 
	//数据转换
	pLedDat    = &ValRGB[0];
	pLedPwmBuf = &ValPwmBuff[1];//LED数据从第二字节开始存放
	for (i = 0;i < (LED_NUM * 3);i++)
	{
		Data = *pLedDat;
		for (j = 0;j < 8;j++)
		{
			*ppLedPwmBuf = (Data & 0x80) ? WS281x_1 : WS281x_0;
			pLedPwmBuf++;
			Data <<= 1;
		}
		pLedDat++;
	}
 
	//设置第一和最后数据为0
	ValPwmBuff[0] = 0;
	ValPwmBuff[LEN_LED_BIT - 1] = 0;
 
	//要修改数据传输量及内存、外设地址是,要先让通道不工作
	DMA1_Channel3->CCR  &= ~DMA_CCR3_EN;
	//设置传输的数据量
	DMA1_Channel3->CNDTR = LEN_LED_BIT;
	//设置外设地址
	DMA1_Channel3->CPAR  = (uint32_t)&TIM3->CCR4;
	//设置存储器地址
	DMA1_Channel3->CMAR  = (uint32_t)&LedRGB.LedBuff[0];
	//通道优先级最高,存储器/外设数据宽度16位,存储器地址自增,从存储器读数据,通道3开启
	DMA1_Channel3->CCR   = DMA_CCR3_PL + DMA_CCR3_MSIZE_0 + DMA_CCR3_PSIZE_0 +
		                   DMA_CCR3_MINC + DMA_CCR3_DIR + DMA_CCR3_EN;

}

程序代码就结束了。使用时先调用WS281xInit()函数初始相关模块,然后调用SetRGBData函数设置LED灯的值,最后调用WS281xSend()函数发送数据。

下图是我程序发送4个灯的波形:

发表评论