GD32F470VET6使用SPI+DMA操作ws2812B灯带

目录

1、使用引脚

2、变量定义

3、GPIO初始化

4、SPI初始化

5、DMA初始化

​6、初始化

7、设定某个RGB LED的颜色

8、DMA发送数据

9、设置红灯


1、使用引脚

PA7  对应SPI0_Tx, 使用DMA1的通道5

2、变量定义

#define LED_GPIO_RCU     RCU_GPIOA
#define LED_GPIO_PERIPH  GPIOA
#define LED_GPIO_PIN     GPIO_PIN_7
#define LED_ALT_NUM      GPIO_AF_5   // 复用

#define LED_SPI_RCU      RCU_SPI0
#define LED_SPI          SPI0

#define LED_DMA_RCU      RCU_DMA1
#define LED_DMA          DMA1
#define LED_DMA_CHANNEL  DMA_CH5
#define LED_DMA_SUBPERI  DMA_SUBPERI3

#define CODE0 0xE0 // 0码, 发送的时间 1110 0000  根据不同的SCK适当调整
#define CODE1 0xF8 // 1码, 发送的时间 1111 1000

const RGBColor_TypeDef RED      = {255,0,0};
const RGBColor_TypeDef GREEN    = {0,255,0};
const RGBColor_TypeDef BLUE     = {0,0,255};

const uint16_t PIXEL_NUM = 8;       // 灯珠 RGB数量
uint8_t pixelBuffer[PIXEL_NUM][24] ;// 灯珠

3、GPIO初始化

void AlarmLed_GPIOInit(void)
{
    rcu_periph_clock_enable(LED_GPIO_RCU);   /* 使能 GPIOA的时钟 */
    gpio_af_set            (LED_GPIO_PERIPH, LED_ALT_NUM,   LED_GPIO_PIN);
    gpio_mode_set          (LED_GPIO_PERIPH, GPIO_MODE_AF,  GPIO_PUPD_NONE,    LED_GPIO_PIN);
    gpio_output_options_set(LED_GPIO_PERIPH, GPIO_OTYPE_PP, GPIO_OSPEED_50MHZ, LED_GPIO_PIN);
}

4、SPI初始化

void AlarmLed_SPIInit(void)
{
    rcu_periph_clock_enable(RCU_SPI0);    /* 使能 SPI的时钟 */
    
    /* SPI 配置 */
    spi_disable(LED_SPI);
    spi_parameter_struct spi_init_struct;
    spi_init_struct.trans_mode           = SPI_TRANSMODE_FULLDUPLEX; /* 设置传输模式 */
    spi_init_struct.device_mode          = SPI_MASTER;               /* 设置工作模式 */
    spi_init_struct.frame_size           = SPI_FRAMESIZE_8BIT;       /* 设置SPI的数据大小:SPI发送接收8位帧结构 */
    spi_init_struct.clock_polarity_phase = SPI_CK_PL_LOW_PH_2EDGE;   /* 串行同步时钟的空闲状态为低电平 */
    spi_init_struct.nss                  = SPI_NSS_SOFT;             /* nss 软件控制 */
    spi_init_struct.prescale             = SPI_PSC_16;               /* 定义波特率预分频的值 */
    spi_init_struct.endian               = SPI_ENDIAN_MSB;           /* 传输方式大端字节 */
    spi_init(LED_SPI, &spi_init_struct);  /* 初始化 */
    spi_enable(LED_SPI);                  /* 使能 SPI */
    
    /* SPI/I2S 配置 */
    spi_dma_enable(LED_SPI, SPI_DMA_TRANSMIT);
}

WS2812B编码协议(单位:ns):
                   min     typ     max
       bit 0
         T0H:  220      -      420
         T0L:  750      -      1600
        
       bit 1 
         T1H:  750      -      1600
          T1L:  220      -      420

       RESET: time > 300us

SPI0 --> APB2 时钟 120MHz

16分频APB2,120MHz/16 = 7.5MHz
时钟周期为:1/7.5/1e6 = 1.33e-7 = 133ns
     
0码: 111(3*133=399)0 0000(5*133=665)   --> 0xE0
1码: 1111 1(5*133=665) 000(3*133=399)  --> 0xF8

算出的时间与手册的略微有点出入,但是经过测试,SPI使用16分频,0码为0xE0,1码为0xF8,灯光显示最为稳定

5、DMA初始化

void AlarmLed_DMAInit(void)
{
    rcu_periph_clock_enable(LED_DMA_RCU);    /* 使能 SPI的时钟 */
    
    dma_single_data_parameter_struct dma_init_SPI0_TX={0};
    
    /* DMA 配置 */
    dma_deinit(LED_DMA, LED_DMA_CHANNEL);      /* SPI0_Tx --> DMA1 Channel5 */
    dma_init_SPI0_TX.periph_addr = (uint32_t)(&SPI_DATA(LED_SPI));  /* DMA外设ADC基地址 */
    dma_init_SPI0_TX.periph_inc  = DMA_PERIPH_INCREASE_DISABLE;     /* 外设地址寄存器不变 */
    dma_init_SPI0_TX.memory0_addr= (uint32_t)pixelBuffer;           /* DMA内存基地址 */
    dma_init_SPI0_TX.memory_inc  = DMA_MEMORY_INCREASE_ENABLE;      /* 内存地址寄存器递增 */
    dma_init_SPI0_TX.periph_memory_width = DMA_PERIPH_WIDTH_8BIT;   /* 外设数据传输宽度 */
    dma_init_SPI0_TX.direction   = DMA_MEMORY_TO_PERIPH;            /* 数据传输方向,从内存读取发送到外设 */
    dma_init_SPI0_TX.number      = PIXEL_NUM * 24;                  /* DMA通道的DMA缓存的大小 */
    dma_init_SPI0_TX.priority    = DMA_PRIORITY_HIGH;               /* DMA通道拥有中优先级  */
    dma_init_SPI0_TX.circular_mode = DMA_CIRCULAR_MODE_DISABLE; 
    dma_single_data_mode_init(LED_DMA, LED_DMA_CHANNEL, &dma_init_SPI0_TX);
    dma_channel_subperipheral_select(LED_DMA, LED_DMA_CHANNEL, DMA_SUBPERI3);
    dma_channel_enable(LED_DMA, LED_DMA_CHANNEL);
}

需要注意的是

dma_channel_subperipheral_select(LED_DMA, LED_DMA_CHANNEL, DMA_SUBPERI3);

该函数的第三个参数是通道外设的值

6、初始化

void AlarmLed_Init()
{
    AlarmLed_GPIOInit();
    AlarmLed_SPIInit();
    AlarmLed_DMAInit();
}

7、设定某个RGB LED的颜色

void rgb_SetColor(uint16_t LedId, RGBColor_TypeDef Color)
{
    uint16_t i;

    if( LedId > ( PIXEL_NUM ) )
    {
        return;
    }

    for(i=0;i<=7;i++)
    {
        pixelBuffer[LedId][i]= ( (Color.G & (1 << (7 -i)) )? (CODE1):CODE0 );
    }
    for(i=8;i<=15;i++)
    {
        pixelBuffer[LedId][i]= ( (Color.R & (1 << (15-i)) )? (CODE1):CODE0 );
    }
    for(i=16;i<=23;i++)
    {
        pixelBuffer[LedId][i]= ( (Color.B & (1 << (23-i)) )? (CODE1):CODE0 );
    }
}

8、DMA发送数据

void rgb_SendArray(void)
{
    if(dma_flag_get(LED_DMA, LED_DMA_CHANNEL, DMA_FLAG_FTF) != RESET)          /* 等待DMA传输完成 */
    { 
        dma_flag_clear(LED_DMA, LED_DMA_CHANNEL, DMA_FLAG_FTF);                /* 清除传输完成标志 */ 
        dma_channel_disable(LED_DMA, LED_DMA_CHANNEL);                         /* 关闭DMA传输 */ 
        
        dma_transfer_number_config(LED_DMA, LED_DMA_CHANNEL, PIXEL_NUM * 24);  /* 数据传输量  */  
        dma_channel_enable(LED_DMA, LED_DMA_CHANNEL);                          /* 开启DMA传输 */  
    }
}

9、设置红灯

void RGB_RED(uint16_t Pixel_LEN)
{
    uint16_t i;
    for(i = 0; i < Pixel_LEN; i++)
    {  
        rgb_SetColor(i,RED);
    }
    rgb_SendArray();
}

  • 0
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
以下是使用GD32F470zit6进行SPI中断接收数据的基本流程: 1. 初始化SPI控制器。设置SPI的通信参数,如时钟极性、相位、数据位、帧格式等。 2. 配置SPI接收缓冲区。开辟一个缓冲区用于存储接收到的数据。 3. 配置SPI中断。使能SPI中断,并设置中断优先级。 4. 在中断服务程序中接收数据。当SPI接收到数据时,会触发中断。在中断服务程序中,读取SPI数据寄存器中的数据,并存储到接收缓冲区中。 下面是一个简单的示例代码,展示如何使用GD32F470zit6进行SPI中断接收数据: ``` #include "gd32f4xx.h" #define SPI_SLAVE_SELECT_PIN GPIO_PIN_4 #define SPI_SLAVE_SELECT_PORT GPIOC uint8_t rx_buffer[256]; volatile uint8_t rx_index = 0; void spi_init(void) { spi_parameter_struct spi_init_struct; rcu_periph_clock_enable(RCU_GPIOC); rcu_periph_clock_enable(RCU_SPI1); gpio_init(SPI_SLAVE_SELECT_PORT, GPIO_MODE_OUT_PP, GPIO_OSPEED_50MHZ, SPI_SLAVE_SELECT_PIN); spi_i2s_deinit(SPI1); spi_init_struct.trans_mode = SPI_TRANSMODE_RECEIVEONLY; spi_init_struct.device_mode = SPI_DEVICE_SLAVE; spi_init_struct.frame_size = SPI_FRAMESIZE_8BIT; spi_init_struct.clock_polarity_phase = SPI_CK_PL_LOW_PH_2EDGE; spi_init_struct.nss = SPI_NSS_HARD; spi_init_struct.prescale = SPI_PSC_256; spi_init_struct.endian = SPI_ENDIAN_LSB; spi_init(SPI1, &spi_init_struct); nvic_irq_enable(SPI1_IRQn, 0, 0); spi_i2s_interrupt_enable(SPI1, SPI_I2S_INT_RBNE); } void SPI1_IRQHandler(void) { if (RESET != spi_i2s_interrupt_flag_get(SPI1, SPI_I2S_INT_RBNE)) { rx_buffer[rx_index++] = spi_i2s_data_receive(SPI1); spi_i2s_interrupt_flag_clear(SPI1, SPI_I2S_INT_RBNE); } } int main(void) { spi_init(); while(1) { // Wait for data to be received if (rx_index > 0) { // Process received data // ... // Reset receive buffer index rx_index = 0; } } } ``` 在这个示例中,我们使用SPI1作为从设备(SPI_DEVICE_SLAVE),并开启接收中断。当SPI接收到数据时,会触发中断,中断服务程序中会将接收到的数据存储到rx_buffer中,并将rx_index加1。在主程序中,我们可以通过检查rx_index是否大于0来判断是否有数据接收,然后进行相应的处理。注意,在处理完数据后,需要将rx_index重置为0。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

lucky_sunshine

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

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

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

打赏作者

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

抵扣说明:

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

余额充值