SPI+DMA驱动和控制WS2812彩色RGB灯
SPI+DMA驱动WS2812
比赛的能量机关,官方的灯条是SK6812灯珠,每米144个灯珠,但是真的贵(
最后还是选用了WS2811灯珠。
很恶心的一点是,WS2811的逻辑0和1表达示方式这部分,网上好多图都是不对的,一定要以数据手册为准… 如果逻辑0和逻辑1的发送时序不对,那么整个灯条的颜色都是乱的甚至根本不会亮
先放一段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);
}
}