STM32驱动WS2812

WS2812是一种内部集成驱动电路的RGB全彩灯,引脚如下

灯可以串联首尾相接,由一根信号线进行控制。

完成GPIO管脚初始化之后,可以直接调用RGB_LED_Write_24Bits(uint8_t green,uint8_t red,uint8_t blue)函数进行控制,也可以通过for循环来驱动更多的灯进行色彩显示。

控制程序如下:

#include "ws2818b.h"


void RGB_LED_Init(void)
{
	GPIO_InitTypeDef  GPIO_InitStructure;
 	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);	 
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;				 
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; 		 
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;		
	GPIO_Init(GPIOA, &GPIO_InitStructure);					
	GPIO_SetBits(GPIOA,GPIO_Pin_8);						 
}

/********************************************************/
//
/********************************************************/
void RGB_LED_Write0(void)
{
	RGB_LED_HIGH;
	__nop();__nop();__nop();__nop();__nop();__nop();
	RGB_LED_LOW;
	__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();
	__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();
	__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();
	__nop();__nop();
}

/********************************************************/
//
/********************************************************/

void RGB_LED_Write1(void)
{
	RGB_LED_HIGH;
	__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();
	__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();
	__nop();__nop();
	RGB_LED_LOW;
	__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();__nop();
	__nop();__nop();
}

void RGB_LED_Reset(void)
{
	RGB_LED_LOW;
	delay_us(80);
}

void RGB_LED_Write_Byte(uint8_t byte)
{
	uint8_t i;

	for(i=0;i<8;i++)
		{
			if(byte&0x80)
				{
					RGB_LED_Write1();
			}
			else
				{
					RGB_LED_Write0();
			}
		byte <<= 1;
	}
}

void RGB_LED_Write_24Bits(uint8_t green,uint8_t red,uint8_t blue)
{
	RGB_LED_Write_Byte(green);
	RGB_LED_Write_Byte(red);
	RGB_LED_Write_Byte(blue);
}




void RGB_LED_Red(void)
{
	 uint8_t i;
	for(i=0;i<1;i++)
		{
			RGB_LED_Write_24Bits(0, 0xff, 0);
	}
}

void RGB_LED_Green(void)
{
	uint8_t i;

	for(i=0;i<1;i++)
		{
			RGB_LED_Write_24Bits(0xff, 0, 0);
	}
}

void RGB_LED_Blue(void)
{
	uint8_t i;

	for(i=0;i<1;i++)
		{
			RGB_LED_Write_24Bits(0, 0, 0xff);
	}
}

头文件如下:

#ifndef __WS2818B_H
#define __WS2818B_H	 
#include "sys.h"


#define 	RGB_LED 	GPIO_Pin_8//GPIO¿ÚÊä³ö¶Ë¿Ú
#define		RGB_LED_HIGH	(GPIO_SetBits(GPIOA,RGB_LED))
#define 	RGB_LED_LOW		(GPIO_ResetBits(GPIOA,RGB_LED))

void RGB_LED_Init(void);
void RGB_LED_Write0(void);
void RGB_LED_Write1(void);
void RGB_LED_Reset(void);
void RGB_LED_Write_Byte(uint8_t byte);
void RGB_LED_Write_24Bits(uint8_t green,uint8_t red,uint8_t blue);
void RGB_LED_Red(void);
void RGB_LED_Green(void);
void RGB_LED_Blue	(void);



#endif

 

  • 15
    点赞
  • 117
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
驱动WS2812 LED灯条,需要使用STM32的定时器和DMA功能。以下是一个简单的示例代码: ``` #include "stm32f10x.h" #define LED_NUM 8 uint8_t led_data[LED_NUM * 3]; void WS2812_Init(void) { // 配置定时器 TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct; TIM_OCInitTypeDef TIM_OCInitStruct; GPIO_InitTypeDef GPIO_InitStruct; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB | RCC_APB2Periph_TIM1, ENABLE); GPIO_InitStruct.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_InitStruct.GPIO_Pin = GPIO_Pin_13; GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOB, &GPIO_InitStruct); TIM_TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInitStruct.TIM_Period = 89; TIM_TimeBaseInitStruct.TIM_Prescaler = 0; TIM_TimeBaseInit(TIM1, &TIM_TimeBaseInitStruct); TIM_OCInitStruct.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStruct.TIM_OutputState = TIM_OutputState_Enable; TIM_OCInitStruct.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OCInitStruct.TIM_Pulse = 0; TIM_OC1Init(TIM1, &TIM_OCInitStruct); TIM_Cmd(TIM1, ENABLE); // 配置DMA DMA_InitTypeDef DMA_InitStruct; RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); DMA_DeInit(DMA1_Channel2); DMA_InitStruct.DMA_BufferSize = LED_NUM * 24; DMA_InitStruct.DMA_DIR = DMA_DIR_PeripheralDST; DMA_InitStruct.DMA_M2M = DMA_M2M_Disable; DMA_InitStruct.DMA_MemoryBaseAddr = (uint32_t)led_data; DMA_InitStruct.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte; DMA_InitStruct.DMA_MemoryInc = DMA_MemoryInc_Enable; DMA_InitStruct.DMA_Mode = DMA_Mode_Normal; DMA_InitStruct.DMA_PeripheralBaseAddr = (uint32_t)&TIM1->CCR1; DMA_InitStruct.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte; DMA_InitStruct.DMA_PeripheralInc = DMA_PeripheralInc_Disable; DMA_InitStruct.DMA_Priority = DMA_Priority_High; DMA_Init(DMA1_Channel2, &DMA_InitStruct); // 发送复位信号 uint8_t reset_data[4] = {0x00, 0x00, 0x00, 0x00}; DMA_Cmd(DMA1_Channel2, ENABLE); DMA_SetCurrDataCounter(DMA1_Channel2, 4); DMA_MemoryTargetConfig(DMA1_Channel2, (uint32_t)reset_data, DMA_Memory_0); while (DMA_GetFlagStatus(DMA1_FLAG_TC2) == RESET); DMA_ClearFlag(DMA1_FLAG_TC2); } void WS2812_SendData(void) { DMA_Cmd(DMA1_Channel2, ENABLE); DMA_SetCurrDataCounter(DMA1_Channel2, LED_NUM * 24); DMA_MemoryTargetConfig(DMA1_Channel2, (uint32_t)led_data, DMA_Memory_0); while (DMA_GetFlagStatus(DMA1_FLAG_TC2) == RESET); DMA_ClearFlag(DMA1_FLAG_TC2); } void WS2812_SetColor(uint8_t index, uint8_t r, uint8_t g, uint8_t b) { uint32_t color = ((uint32_t)g << 16) | ((uint32_t)r << 8) | (uint32_t)b; uint32_t mask = 0x800000; for (int i = 0; i < 24; i++) { if (color & mask) { led_data[index * 24 + i] = 18; } else { led_data[index * 24 + i] = 9; } mask >>= 1; } } int main(void) { WS2812_Init(); while (1) { for (int i = 0; i < LED_NUM; i++) { WS2812_SetColor(i, 255, 0, 0); } WS2812_SendData(); delay_ms(500); for (int i = 0; i < LED_NUM; i++) { WS2812_SetColor(i, 0, 255, 0); } WS2812_SendData(); delay_ms(500); for (int i = 0; i < LED_NUM; i++) { WS2812_SetColor(i, 0, 0, 255); } WS2812_SendData(); delay_ms(500); } } ``` 在这个例子中,我们使用了定时器1的通道1来控制WS2812的数据线。通过配置定时器的周期和占空比,可以产生0和1的高电平时间。然后,我们使用DMA将数据发送到定时器的CCR1寄存器,以控制高电平时间的长度。 在WS2812_SendData函数中,我们启动DMA传输,将led_data数组中的数据发送到WS2812灯条。在WS2812_SetColor函数中,我们将颜色值转换为对应的高电平时间,存储在led_data数组中。 需要注意的是,在初始化之前,需要使用GPIO_ResetBits将数据线拉低一段时间,以产生复位信号。在发送数据之后,需要等待一段时间,以确保数据已经全部发送完毕。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值