SPI+DMA驱动和控制WS2812彩色RGB灯

SPI+DMA驱动和控制WS2812彩色RGB灯

SPI+DMA驱动WS2812

比赛的能量机关,官方的灯条是SK6812灯珠,每米144个灯珠,但是真的贵(

最后还是选用了WS2811灯珠。

很恶心的一点是,WS2811的逻辑0和1表达示方式这部分,网上好多图都是不对的,一定要以数据手册为准… 如果逻辑0和逻辑1的发送时序不对,那么整个灯条的颜色都是乱的甚至根本不会亮

SPI+DMA驱动和控制WS2812彩色RGB灯_第1张图片

SPI+DMA驱动和控制WS2812彩色RGB灯_第2张图片

ws28123

先放一段QA吧

Q1: 像这种一根信号线输出控制的逻辑电平一般都要求5V及以上,需要3.3V升5V吗

A1: STM32虽然只能输出3.3V,但是控WS2811是可以的。另外3.3V升到5V,三极管或是MOS管的信号上升时间相对而言太长了,跟不上控制信号的变化,导致几乎输不出信号

Q2: 直接用IO口翻转可以控制吗

A2: 不太行。用封装库函数翻转IO的执行速度,很难满足几百ns到1us级别的时间要求,虽然可以直接操作寄存器,但即使是对着示波器一点点测试NOP空指令的个数,也只是一种时序近似。随着灯条长度的加长和信号发送时间的推移,总会出现逻辑0和1的误识别的情况

Q3: FreeRTOS 会影响控制的时序吗

A3: 配上稍高一些的任务优先级,使用SPI+DMA不会影响

Q4: 一条总线能控多少灯

A4: 测试一条SPI+DMA控了1920个灯珠也依旧能正常控制,更多的灯珠没设备了也就没测。只要时序对了理论上是能控制无限个的

控制方式

单总线,通过总线上高低电平时间长短的不同来区分0和1。

WS2811的工作频率为800KHz,如果将SPI的频率设置为6.4MHz,正好是灯条IC芯片的8倍,那么每一个字节(8位)正好对应一个逻辑位,也就是11110000代表逻辑1,11000000代表逻辑0

不过协议也有150ns的兼容性。比如SPI设置为5.25MBits/s(Mbps)(84MHz时钟线16分频据线是8Bits),一Bit的发送时间为1/5.25=190ns一个字节约1.52us。所以可以通过SPI的MOSI输出信号线发送一个字节Byte(8Bits)的数据模拟 WS2812的一个位。所以发送一个uint8字节0xF8(11111000),高电平时间为5x190=950ns在580ns-1us范围之间,代表逻辑1;0x80(10000000),高电平时间为190ns接近220ns-380ns范围,代表逻辑0

CubeMX 配置

  • 开启 SPI1,因为只用到了MOSI输出信号线,所以模式选择Transmit Only Master 仅发送主模式 就可以。

    • Data Size 设置为 8 Bits。理由是因为上面的控制方式

    • CPHA 设置为 2 Edge。跳变沿为1时,MOSI空闲电平为高电平。跳变沿设置为2时,会延续上次发送的最后电平,因为发送数据的末尾都是低电平,这样 WS2812 不会误判

    • CPOL 设置为 High。总线空闲时SCK的时钟状态为高电平

  • 开启 DMA。Mode设置为Normal,由程序触发DMA发送即可。

程序

#define LED_NUM 64	// LED灯珠个数

// 模拟bit码: 0x80为逻辑0, 0xF8为逻辑1
const uint8_t code[]={0x80,0xF8};

typedef struct {
  uint8_t R;
  uint8_t G;
  uint8_t B;
}RGBColor_TypeDef;	//颜色结构体

// 常见颜色定义
const RGBColor_TypeDef RED = {255, 0, 0};
const RGBColor_TypeDef BLUE = {0, 0, 255};

// 灯颜色缓存
RGBColor_TypeDef RGB_Data[LED_NUM] = {0};

/**
  * @brief			设置灯带颜色发送缓存
  * @param[in]		ID 颜色
  */
void Set_LEDColor(uint16_t LedId, RGBColor_TypeDef Color)
{
    RGB_Data[LedId].G = Color.G;
    RGB_Data[LedId].R = Color.R;
    RGB_Data[LedId].B = Color.B;
}

/**
  * @brief			SPI发送控制ws2812
  * @param[in]		待发送缓存
  */
static void SPI_Send_WS2812(uint8_t *SPI_RGB_BUFFER)
{
    // 判断上次DMA有没有传输完成
    while(HAL_DMA_GetState(&hdma_spi1_tx) != HAL_DMA_STATE_READY);
    // 发送一个24bit的RGB数据
    HAL_SPI_Transmit_DMA(&hspi1, SPI_RGB_BUFFER, 24);
}

/**
  * @brief			控制WS2812
  * @param[in]		待发送缓存
  */
void RGB_Reflash(void)
{
	static uint8_t RGB_BUFFER[24]={0};
	uint8_t dat_b,dat_r,dat_g;
	// 将数组颜色转化为24个要发送的字节数据
    for (uint16_t i = 0; i < LED_NUM; i++) {
        dat_g = RGB_DAT[i].G;
        dat_r = RGB_DAT[i].R;
        dat_b = RGB_DAT[i].B;
        for (uint16_t j = 0; j < 8; j++) {
            RGB_BUFFER[7-j] =code[dat_g & 0x01];
            RGB_BUFFER[15-j]=code[dat_r & 0x01];
            RGB_BUFFER[23-j]=code[dat_b & 0x01];
            dat_g >>=1;
            dat_r >>=1;
            dat_b >>=1;
        }
        SPI_Send_WS2812(RGB_BUFFER);
	}
}

你可能感兴趣的:(STM32,RoboMaster,单片机,stm32,嵌入式硬件)