gaoyang9992006 发表于 2024-8-1 10:26

使用MCC配置SPI硬件驱动WS2812B灯环

本帖最后由 gaoyang9992006 于 2024-8-1 16:53 编辑

#申请原创#
关于MCC提速的问题查看下面连接的帖子
https://bbs.21ic.com/icview-3393632-1-1.html
接下来一步一步从新建工程讲起
1、新建工程

2、选择MCC Melody

3、选择本次要使用的设备资源SPI_Host和DELAY


4、接下来对时钟进行配置,将SPI时钟速率设置为8MHz
点击左边Project Resources中的System下的Clock Control,右边会弹出上面界面选项卡,将Clock Source设置为HFINTOSC。
HF Internal Clock 选择为16Mhz,这样可以方便分频出8MHz,
Clock Divider 设置为1,主系统时钟不分频
点击左边Project Resources中的Drivers下的SPI_Host,在右边的设置选项卡做如下图中的设置,将Requested Speed(kHz)设置为8000
即设置到了8Mhz。这样脉宽的最小单位就是0.125us了。

5、接下来需要对IO选择,由哪个IO接口输出SPI数据。引脚的选择是有讲究的,不可随意指定引脚使用,首先你要确定该引脚是否被其他功能占用,这个要根据你的板子上的设计来看,如果是全新的设计,那么选择会更加自有一些。
这里要打开芯片的用户手册,找到引脚分配表一章节,本次要使用的是SPI,另外根据选择的元件封装找到对应的表格。


我们注意到这些备注为1的引脚的默认功能是指定了的,根据备注,可知这是一个可重新映射的PPS输入信号。输入函数可以从显示的默认位置移动到其他几个PORTx引脚之一。
其他横线的引脚为备注2的,在这些行中显示的所有数字输出信号都是PPS可重新映射的。这些信号可以映射到输出到几个PORTx引脚选项之一。我使用的是PIC18F16QQ41-CNANO开发板,因此应按照该开发板原理图,找到合适没有被占用的引脚来使用,我使用的引脚配置如下图所示:


6、驱动原理
要知道驱动原理,需要先知道WS2812B的驱动时序

每一个LED由24BIT数据驱动

数据的时序波形如下图所示

数据参数的定义

LED 灯珠主要根据高电平时间判断“0”码和“1”码。
高电平时间介于 200ns~410ns,IC 判断为“0”码,高电平时间介于 640ns~1000ns,判断为“1”码。
“0”码和“1”码的低电平代表此码结束,准备接收下一数据码。
由于我们配置的SPI时钟为8MHz,那么每一个脉宽为0.125us
对于“0”码的高电平时间,2个脉宽为0.25us即可满足要求,因此我们用具备2个脉宽高电平的数据表示0,这个数据就是:0xC0
对于“1”码的高电平时间,6个脉宽为0.75us即可满足要求,因此我们用具备6个脉宽高电平的数据表示1,这个数据就是:0xFC
明白了这个原理,就可以编写代码了。
7、代码编写
typedef struct COLOR
{
    uint8_t redChannel;
    uint8_t greenChannel;
    uint8_t blueChannel;
} color_t;

void ws2812_send(unsigned char x)
{
    for(int i=0;i<8;i++)
    {
      x<<=i;
      if(x&0x80)
      {
            SPI1_ByteWrite(0xFC);
            while(SPI1_IsTxReady()==false);
      }
      else
      {
            SPI1_ByteWrite(0xC0);
            while(SPI1_IsTxReady()==false);
      }
    }
}

void WriteNeopixel(color_t const color)
{
    ws2812_send(color.greenChannel);   
    ws2812_send(color.redChannel);   
    ws2812_send(color.blueChannel);
}另外我们只使用SDO的发送数据功能,因此我们要对函数SPI1_ByteWrite进行修改一下。
找到spi1.c文件
将函数修改为
void SPI1_ByteWrite(uint8_t byteData)
{      
      SPI1CON2 = (SPI1CON2 & ~(_SPI1CON2_SPI1RXR_MASK)) | (_SPI1CON2_SPI1TXR_MASK);
      SPI1TCNTL = 1;
      //Write input data to SPI transmit buffer register
      SPI1TXB = byteData;
}即对SPI1CON2只设置为发送模式。
然后在main函数就可以使用上述函数对灯珠进行控制了。

#include "mcc_generated_files/system/system.h"
#include"mcc_generated_files/timer/delay.h"

#include"ws2812b.h"


//显示缓存,所有的变化操作都在这个数组进行
long led_buff_temp={0};

/*
    Main application
*/
color_t neopixelsBuffer = {

    { 64, 100, 64 }, { 64, 64,0 }, {0, 64, 64 }, {0, 64,0 },

    { 64,0, 64 }, { 64,0,0 }, {0,0, 64 }, {30,10,2 },

    { 96, 96, 96 }, { 48, 48, 48 }, { 24, 24, 24 }, { 12, 24, 12 },

    {36,36,6 }, {3,33,33 }, {16,16,1 }, {50,0,0 },

    {6,60,6 }, {3,3,30 }, {10,1,10 }, {0,10,0 }

};
color_t temp_led;

int main(void)
{
    SYSTEM_Initialize();
    // If using interrupts in PIC18 High/Low Priority Mode you need to enable the Global High and Low Interrupts
    // If using interrupts in PIC Mid-Range Compatibility Mode you need to enable the Global Interrupts
    // Use the following macros to:

    // Enable the Global Interrupts
    //INTERRUPT_GlobalInterruptEnable();

    // Disable the Global Interrupts
    //INTERRUPT_GlobalInterruptDisable();
    SPI1_Open(HOST_CONFIG);
//点亮
    for(int i=0;i<20;i++) WriteNeopixel(neopixelsBuffer);
//色彩滚动   


    while(1)
    {
      temp_led.blueChannel =neopixelsBuffer.blueChannel;
      temp_led.greenChannel=neopixelsBuffer.greenChannel;
      temp_led.redChannel=neopixelsBuffer.redChannel;
      for(int i=0;i<19;i++)
      {
            neopixelsBuffer.blueChannel=neopixelsBuffer.blueChannel;
            neopixelsBuffer.greenChannel=neopixelsBuffer.greenChannel;
            neopixelsBuffer.redChannel=neopixelsBuffer.redChannel;
      }
      neopixelsBuffer.blueChannel =temp_led.blueChannel;
      neopixelsBuffer.greenChannel=temp_led.greenChannel;
      neopixelsBuffer.redChannel=temp_led.redChannel;   
      DELAY_milliseconds(500);
      for(int i=0;i<20;i++) WriteNeopixel(neopixelsBuffer);
    }   
}烧录程序,可以看到灯环上的20个彩色LED颜色滚动起来了。

效果
https://www.bilibili.com/video/BV1BCvbefER3?t=8.3
工程文件下载**** Hidden Message *****







gejigeji521 发表于 2024-8-1 10:50

看起来很不错,学习一下,看看附件。

pzsh 发表于 2024-8-1 11:05

很好的实现方式。 驱动WS2812, 是不是也可以使用CLC, 或者CLB (PIC6F31xx 系列上的)来实现? 这样的话,代码量会少一些。

KCCHEN 发表于 2024-8-1 11:16

看起来很不错,学习一下

gaoyang9992006 发表于 2024-8-1 11:42

pzsh 发表于 2024-8-1 11:05
很好的实现方式。 驱动WS2812, 是不是也可以使用CLC, 或者CLB (PIC6F31xx 系列上的)来实现? 这样的话 ...

嗯,好主意,我看看怎么用CLC实现。

jcy8508 发表于 2024-8-11 00:21

驱动WS2812学习一下

wuyushan 发表于 2024-8-11 11:48

#申请原创#

xinxianshi 发表于 2024-8-13 22:40

手把手教学,懂了,一步一步操作,就觉得挺容易的。看来不能省略一些细节步骤。

AloneKaven 发表于 2024-8-15 23:07

这个整挺好

wanduzi 发表于 2024-9-3 16:00

SPI实现的原理看懂了,如果使用PWM该如何实现呢,如何控制PWM就生成1个周期的脉宽?这一点不懂,没操作过

clolcj 发表于 2024-10-15 23:29

参考

yellow555 发表于 2024-10-31 13:35

脉宽的最小单位是0.125us是为了做什么啊?

gaoyang9992006 发表于 2024-10-31 20:36

yellow555 发表于 2024-10-31 13:35
脉宽的最小单位是0.125us是为了做什么啊?

方便匹配这个通信协议的高低电平长度。

643757107 发表于 2024-11-27 20:43

下载学习参靠一下。

小明的同学 发表于 2024-11-28 10:35

这个灯珠挺好用的。

小明的同学 发表于 2024-11-28 10:35

停止输出后是不是可以保持?

zhuotuzi 发表于 2024-11-28 20:38

火过一段时间的元件,现在好多彩色灯光用的这个。

xixi2017 发表于 2024-11-29 16:25

有没有I2C的操作演示啊,参靠一下。

xixi2017 发表于 2024-11-29 16:25

我想用I2C驱动OLED单色屏。

yiyigirl2014 发表于 2024-11-29 19:51

多少个LED都可以驱动吗
页: [1] 2 3
查看完整版本: 使用MCC配置SPI硬件驱动WS2812B灯环