OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 6010|回复: 2

使用DMA方式传输SPI接口OLED显示数据(附代码)

[复制链接]

3

主题

12

帖子

0

精华

初级会员

Rank: 2

积分
174
金钱
174
注册时间
2015-4-11
在线时间
31 小时
发表于 2021-1-19 21:27:01 | 显示全部楼层 |阅读模式
因为每次用CPU去刷新OLED,会使得CPU大量的工作用于与OLED屏幕的数据传输上,明明是一个很简单的过程却百倍浪费不少CPU资源。于是萌生了用DMA方式将显示数据传输至OLED屏幕。
使用的OLED是某宝上很常见的中景园的7针0.96寸OLED。
(接触STM32不久,代码中有不清晰的地方还请各位大佬见谅)
SPI代码部分:
  1. #include "spi.h"

  2. /**
  3. *        SPI时钟线:GPIOA5
  4. *        SPI数据线:GPIOA7
  5. *        SPI片选线:GPIOA4,使用硬件方式
  6. *        OLED复位线:GPIOA3
  7. *        OLED命令/数据选择线:GPIOA2
  8. */

  9. void SPI1_Init(void)
  10. {
  11.         GPIO_InitTypeDef  GPIO_InitStructure;
  12.         SPI_InitTypeDef  SPI_InitStructure;

  13.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//使能GPIO时钟
  14.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1,  ENABLE);//SPI1时钟使能

  15.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4|GPIO_Pin_5|GPIO_Pin_6|GPIO_Pin_7;
  16.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  17.         GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  18.         GPIO_Init(GPIOA, &GPIO_InitStructure);//根据设置配置SPI相关GPIO口
  19.         GPIO_SetBits(GPIOA,GPIO_Pin_4|GPIO_Pin_5|GPIO_Pin_6|GPIO_Pin_7);

  20.         SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;//设置SPI单向或者双向的数据模式:SPI设置为双线双向全双工
  21.         SPI_InitStructure.SPI_Mode = SPI_Mode_Master;//设置SPI工作模式:设置为主SPI
  22.         SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;//设置SPI的数据大小:SPI发送接收8位帧结构
  23.         SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;//串行同步时钟的空闲状态为高电平
  24.         SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;//串行同步时钟的第二个跳变沿(上升或下降)数据被采样
  25.         SPI_InitStructure.SPI_NSS = SPI_NSS_Hard;//NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制
  26.         SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;//定义波特率预分频的值:波特率预分频值为256
  27.         SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;//指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始
  28.         SPI_InitStructure.SPI_CRCPolynomial = 7;//CRC值计算的多项式
  29.         SPI_Init(SPI1, &SPI_InitStructure);//根据SPI_InitStruct中指定的参数初始化外设SPIx寄存器

  30.         SPI_SSOutputCmd(SPI1, ENABLE);//使能SS位输出使能(因为使用的是硬件NSS信号)
  31.         SPI_Cmd(SPI1, ENABLE); //使能SPI外设
  32. }

  33. //SPI 速度设置函数
  34. //SpeedSet:
  35. //SPI_BaudRatePrescaler_2   2分频   
  36. //SPI_BaudRatePrescaler_8   8分频   
  37. //SPI_BaudRatePrescaler_16  16分频  
  38. //SPI_BaudRatePrescaler_256 256分频
  39. void SPI1_SetSpeed(u8 SPI_BaudRatePrescaler)
  40. {
  41.         assert_param(IS_SPI_BAUDRATE_PRESCALER(SPI_BaudRatePrescaler));
  42.         SPI1->CR1&=0XFFC7;
  43.         SPI1->CR1|=SPI_BaudRatePrescaler;        //设置SPI2速度
  44.         SPI_Cmd(SPI1,ENABLE);

  45. }

  46. //SPIx 读写一个字节
  47. //TxData:要写入的字节
  48. //返回值:读取到的字节
  49. u8 SPI1_ReadWriteByte(u8 TxData)
  50. {               
  51.         u8 retry=0;                                        
  52.         while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET) //检查指定的SPI标志位设置与否:发送缓存空标志位
  53.         {
  54.                 retry++;
  55.                 if(retry>200)
  56.                         return 0;
  57.         }                          
  58.         SPI_I2S_SendData(SPI1, TxData); //通过外设SPIx发送一个数据
  59.         retry=0;
  60.         while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET) //检查指定的SPI标志位设置与否:接受缓存非空标志位
  61.         {
  62.                 retry++;
  63.                 if(retry>200)
  64.                         return 0;
  65.         }                                                              
  66.         return SPI_I2S_ReceiveData(SPI1); //返回通过SPIx最近接收的数据                                            
  67. }
复制代码
DMA代码部分(使用的是DMA循环传输方式,全程无需CPU干预):
  1. #include "dma.h"

  2. /**
  3. * @brief  SPI1使用DMA发送配置函数,内存到外设(SPI1->DR)
  4. * [url=home.php?mod=space&uid=271674]@param[/url]  无
  5. * @retval 无
  6. */
  7. void SPI_DMA_TxConfig(DMA_Channel_TypeDef* DMA_CHx,u32 cpar,u32 cmar,u16 cndtr)
  8. {
  9.         /*DMA初始化结构体*/
  10.         DMA_InitTypeDef DMA_InitStructure;
  11.         /*使能所需时钟*/
  12.         RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);//开启DMA时钟,注意DMA挂载在AHB总线上
  13.         /*将DMA的通道1寄存器重设为缺省值*/
  14.         DMA_DeInit(DMA_CHx);
  15.         /*配置串口3DMA相关参数*/
  16.         DMA_InitStructure.DMA_PeripheralBaseAddr = cpar;//设置DMA外设地址为串口1的DR寄存器地址
  17.         DMA_InitStructure.DMA_MemoryBaseAddr = cmar;//设置内存地址(要传输变量的指针)为数组首地址
  18.         DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;//配置传输方向:内存到外设
  19.         DMA_InitStructure.DMA_BufferSize = cndtr;//配置要传输的数目
  20.         DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;//开启内存地址递增
  21.         DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//禁用外设地址递增
  22.         DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;//配置外设数据单位为一个字节
  23.         DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;//配置内存数据单位为一个字节
  24.         DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;//配置DMA模式为循环
  25.         DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;//配置请求的优先级为中等
  26.         DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;//不开启内存到内存之间的传输
  27.         /*初始化配置DMA*/
  28.         DMA_Init(DMA_CHx, &DMA_InitStructure);//初始化使用到的DMA通道:SPI1的发送对应DMA1的通道3
  29.         /*配置SPI1的DMA发送标志*/
  30.         SPI1->CR2 |= 1<<1;
  31. }
复制代码
OLED.c代码(OLED初始化时要设置成自动地址+1并循环):
  1. #include "oled.h"
  2. #include "delay.h"
  3. #include "spi.h"
  4. #include "dma.h"

  5. /**
  6. *        SPI时钟线:GPIOA5
  7. *        SPI数据线:GPIOA7
  8. *        SPI片选线:GPIOA4,使用硬件方式
  9. *        OLED复位线:GPIOA3
  10. *        OLED命令/数据选择线:GPIOA2
  11. */

  12. u8 DisplayBuff[8][128];

  13. void OLED_Init(void)
  14. {
  15.         GPIO_InitTypeDef  GPIO_InitStructure;
  16.        
  17.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
  18.         /*初始化命令/数据控制引脚及复位引脚*/
  19.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2|GPIO_Pin_3;//GPIOA2:命令数据选择、GPIOA3:复位引脚
  20.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
  21.         GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  22.         GPIO_Init(GPIOA, &GPIO_InitStructure);
  23.         GPIO_SetBits(GPIOA,GPIO_Pin_2|GPIO_Pin_3);
  24.        
  25.         SPI1_Init();//初始化SPI1
  26.         SPI1_SetSpeed(SPI_BaudRatePrescaler_2);//设置SPI1速度
  27.         SPI_DMA_TxConfig(DMA1_Channel3,(u32)(&SPI1->DR),(u32)DisplayBuff,1024);
  28.        
  29.         OLED_Reset();
  30.         OLED_WriteCMD(0xAE);//OLED休眠
  31.         OLED_WriteCMD(0x00);//为页寻址模式设置下栏起始地址
  32.         OLED_WriteCMD(0x10);//为页寻址模式设置更高的列起始地址
  33.         OLED_WriteCMD(0x40);//设置显示起始线(0x00~0x3F)
  34.         OLED_WriteCMD(0x81);//设置对比度控制寄存器
  35.         OLED_WriteCMD(0xCF);//设置SEG输出电流亮度
  36.         OLED_WriteCMD(0xA1);//Set SEG/Column Mapping     0xa0左右反置 0xa1正常
  37.         OLED_WriteCMD(0xC8);//Set COM/Row Scan Direction   0xc0上下反置 0xc8正常
  38.         OLED_WriteCMD(0xA8);//set multiplex ratio(1 to 64)
  39.         OLED_WriteCMD(0x3f);//1/64 duty
  40.         OLED_WriteCMD(0x81);//对比度设置
  41.         OLED_WriteCMD(0x12);//1~255;默认0X7F (亮度设置,越大越亮)
  42.         OLED_WriteCMD(0xD3);//设置显示偏移 (0x00~0x3F)
  43.         OLED_WriteCMD(0x00);//不偏移
  44.         OLED_WriteCMD(0xd5);//时钟频率
  45.         OLED_WriteCMD(0x80);//100 Frames/Sec
  46.         OLED_WriteCMD(0xD9);//预充电持续时间
  47.         OLED_WriteCMD(0xF1);//Set Pre-Charge as 15 Clocks & Discharge as 1 Clock
  48.         OLED_WriteCMD(0xDA);//设置com引脚硬件配置
  49.         OLED_WriteCMD(0x12);//
  50.         OLED_WriteCMD(0xDB);//set vcomh
  51.         OLED_WriteCMD(0x40);//Set VCOM Deselect Level
  52.         OLED_WriteCMD(0x20);//设置内存寻址模式
  53.         OLED_WriteCMD(0x00);//设置成自动巡回扫描模式,先行后纵
  54.         OLED_WriteCMD(0x8D);//设置电荷泵
  55.         OLED_WriteCMD(0x14);//开启电荷泵
  56.         OLED_WriteCMD(0xA4);//Disable Entire Display On (0xa4/0xa5)
  57.         OLED_WriteCMD(0xA6);//正常显示模式,1为亮0为灭
  58.         OLED_WriteCMD(0xAF);//OLED唤醒
  59. }

  60. /*写命令*/
  61. void OLED_WriteCMD(u8 data)
  62. {
  63.         GPIO_ResetBits(GPIOA,GPIO_Pin_2);
  64.         SPI1_ReadWriteByte(data);
  65.         GPIO_SetBits(GPIOA,GPIO_Pin_2);
  66. }
  67. /*写数据*/
  68. void OLED_WriteDATA(u8 data)
  69. {
  70.         GPIO_SetBits(GPIOA,GPIO_Pin_2);
  71.         SPI1_ReadWriteByte(data);
  72.         GPIO_ResetBits(GPIOA,GPIO_Pin_2);
  73. }
  74. /*复位*/
  75. void OLED_Reset(void)
  76. {
  77.         GPIO_ResetBits(GPIOA,GPIO_Pin_3);
  78.         delay_ms(100);
  79.         GPIO_SetBits(GPIOA,GPIO_Pin_3);
  80. }
  81. /*重置下一比特位置到显示屏的起始位*/
  82. void OLED_Start(void)
  83. {
  84.         OLED_WriteCMD (0xb0);//设置页地址(0~7)
  85.         OLED_WriteCMD (0x00);//设置显示位置—列低地址
  86.         OLED_WriteCMD (0x10);//设置显示位置—列高地址
  87.         DMA_Cmd(DMA1_Channel3, ENABLE);//使能SPI1 TX DMA1 所指示的通道
  88. }

  89. /*开启OLED显示,还不完整,若是已经进入DMA传输,请先失能DMA传输再调用此函数*/   
  90. void OLED_Display_On(void)
  91. {
  92.         OLED_WriteCMD(0X8D);//SET DCDC命令
  93.         OLED_WriteCMD(0X14);//DCDC ON
  94.         OLED_WriteCMD(0XAF);//DISPLAY ON
  95. }
  96. /*关闭OLED显示,还不完整,若是已经进入DMA传输,请先失能DMA传输再调用此函数*/
  97. void OLED_Display_Off(void)
  98. {
  99.         OLED_WriteCMD(0X8D);//SET DCDC命令
  100.         OLED_WriteCMD(0X10);//DCDC OFF
  101.         OLED_WriteCMD(0XAE);//DISPLAY OFF
  102. }
复制代码


OLED_DMA.zip

6.94 MB, 下载次数: 447

正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

3

主题

43

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
426
金钱
426
注册时间
2017-1-6
在线时间
66 小时
发表于 2021-3-2 10:55:36 | 显示全部楼层
请问楼主这代码有在实物上跑起来吗,液晶能显示吗?
回复 支持 反对

使用道具 举报

3

主题

16

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
216
金钱
216
注册时间
2018-7-12
在线时间
80 小时
发表于 2021-4-3 17:02:52 | 显示全部楼层
我想知道HAL库 为什么没有 DMA_InitStructure.DMA_PeripheralBaseAddr = cpar;//设置DMA外设地址为串口1的DR寄存器地址
                                             DMA_InitStructure.DMA_MemoryBaseAddr = cmar;//设置内存地址(要传输变量的指针)为数组首地址
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-2-27 08:13

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表