【奈茶君】STM32F407驱动WS2812


一、WS2812概述:

WS2812 内部集成了处理芯片和3颗不同颜色的led灯(红,绿,蓝),通过单总线协议分别控制三个灯的亮度强弱,达到全彩的效果。

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

二、WS2812驱动的几种方式:

1、使用 延时函数
本文采用的驱动方式,可以更换任意引脚。直接翻转IO口产生时序,这种方式最为简单易用,只需要控制延时的时间,就可以从产生0和1码,但是它需要占用系统资源,主要放在主循环中。

2、通过SPI控制
只需要控制在合适的波特率,在传输不同数据的时候,可以产生符合要求的0和1码,这种方式需要等同于使用了一个SPI设备。

3、使用 DMA+Timer 产生时序
这种方式需要使用一个定时器,其中一个通道固定产生一个周期1.25us的PWM,占空比2/3,接着需要另一个通道,在周期的1/3处搬运数据到IO口,若为1,PWM不变,若为0,PWM则为0码,这种方式有更大的局限性,由于DMA只能搬运至少一个字节,所以每次会同时改变8个IO口的高低电平,或许使用位带操作可以解决这问题。

4、使用 Timer+PWM+DMA 产生时序
这种方案有2种驱动的方式,一种是直接建立一个大的数组,存放所有灯珠的数据,然后启动DMA传输,第二种是建立2个灯组数据大小的数组,当DMA传输一个灯珠数据时,改变另一个灯组数据,通过不断改变数组的方式,节约内存,相比较而言,第一种方式较为直观,第二种方式则可以解决灯珠较多的情况。

三、核心代码:

1.C文件:

#include "ws2818b.h"

void RGB_LED_Init(void)
{
    GPIO_InitTypeDef  GPIO_InitStructure;

    RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;//普通输出模式
    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//推挽输出
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOB, &GPIO_InitStructure);
    GPIO_SetBits(GPIOB, GPIO_Pin_8);
}


//一个__nop()延时1/168MHZ   168HMZ为板子主频
void RGB_LED_Write0(void)
{
	RGB_LED_HIGH;	//以下为延时0.4us
	__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();__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();__nop();__nop();
	
	
	RGB_LED_LOW;	//以下为延时0.8us
	__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();__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();__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();__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();__nop();__nop();__nop();__nop();
}

 
void RGB_LED_Write1(void)
{
	RGB_LED_HIGH;	//以下为延时0.8us
	__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();__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();__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();__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();__nop();__nop();__nop();__nop();
	
	
	RGB_LED_LOW;	//以下为延时0.4us
	__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();__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();__nop();__nop();


}
	
void RGB_LED_Reset(void)
{
    RGB_LED_LOW;
    delay_ms(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);
    RGB_LED_Reset();
}

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);
    }
}

2.头文件:

代码如下(示例):

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


#define 	RGB_LED 	GPIO_Pin_8
#define		RGB_LED_HIGH	(GPIO_SetBits(GPIOB,RGB_LED))
#define 	RGB_LED_LOW		(GPIO_ResetBits(GPIOB,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

补充:

介绍下__NOP()函数:
它类似于汇编里的伪指令,也称作“空指令”;
它不执行操作,但占一个程序步。

空指令使用时会占用执行一个指令的CPU时间片。常用于程序延时或精确计时,不过在较快的CPU上不明显。

在本文中由于STM32F407主频达到168MHZ,所以一个__NOP()占用1/168M秒。


  • 19
    点赞
  • 92
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 5
    评论
stm32f407zgt6是一款常用的ARM Cortex-M4内核的微控制器,它具有丰富的外设和强大的性能。在开发过程中,延时函数是常用的一种方法,用于实现一定时间的延时操作。 在stm32f407zgt6上实现延时函数有多种方法,其中比较常用的有以下几种: 1. 使用循环延时:这是最简单的延时方法,通过循环执行空操作来实现延时。例如,以下代码实现了一个大约1毫秒的延时函数: ```c void delay_ms(uint32_t ms) { for (uint32_t i = 0; i < ms * 1000; i++) { __NOP(); } } ``` 2. 使用SysTick定时器:stm32f407zgt6内置了SysTick定时器,可以用于实现精确的延时。首先需要初始化SysTick定时器,然后在延时函数中使用SysTick定时器的计数器进行判断。以下是一个使用SysTick定时器实现的延时函数示例: ```c void delay_ms(uint32_t ms) { SysTick_Config(SystemCoreClock / 1000); // 初始化SysTick定时器为1毫秒 uint32_t start = SysTick->VAL; uint32_t target = start - ms * (SystemCoreClock / 1000); if (target > start) { while (SysTick->VAL > target); } else { while (SysTick->VAL > target && SysTick->VAL < start); } SysTick->CTRL = 0; // 关闭SysTick定时器 } ``` 3. 使用定时器中断:stm32f407zgt6还具有多个定时器,可以使用定时器中断来实现延时。首先需要初始化定时器,并配置一个适当的定时时间。在延时函数中等待定时器中断触发即可。以下是一个使用定时器中断实现的延时函数示例: ```c void delay_ms(uint32_t ms) { TIM_HandleTypeDef htim; htim.Instance = TIMx; // 替换为实际使用的定时器 htim.Init.Prescaler = SystemCoreClock / 1000 - 1; // 设置预分频值,使定时器的计数频率为1kHz htim.Init.Period = ms; // 设置定时器的周期为ms毫秒 HAL_TIM_Base_Init(&htim); HAL_TIM_Base_Start(&htim); while (__HAL_TIM_GET_FLAG(&htim, TIM_FLAG_UPDATE) == RESET); HAL_TIM_Base_Stop(&htim); } ``` 以上是几种常用的stm32f407zgt6延时函数的实现方法,你可以根据自己的需求选择适合的方法来实现延时操作。
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

奈茶君

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

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

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

打赏作者

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

抵扣说明:

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

余额充值