初级会员

- 积分
- 174
- 金钱
- 174
- 注册时间
- 2015-4-11
- 在线时间
- 31 小时
|
因为每次用CPU去刷新OLED,会使得CPU大量的工作用于与OLED屏幕的数据传输上,明明是一个很简单的过程却百倍浪费不少CPU资源。于是萌生了用DMA方式将显示数据传输至OLED屏幕。
使用的OLED是某宝上很常见的中景园的7针0.96寸OLED。
(接触STM32不久,代码中有不清晰的地方还请各位大佬见谅)
SPI代码部分:
- #include "spi.h"
- /**
- * SPI时钟线:GPIOA5
- * SPI数据线:GPIOA7
- * SPI片选线:GPIOA4,使用硬件方式
- * OLED复位线:GPIOA3
- * OLED命令/数据选择线:GPIOA2
- */
- void SPI1_Init(void)
- {
- GPIO_InitTypeDef GPIO_InitStructure;
- SPI_InitTypeDef SPI_InitStructure;
- RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);//使能GPIO时钟
- RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE);//SPI1时钟使能
- GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4|GPIO_Pin_5|GPIO_Pin_6|GPIO_Pin_7;
- GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
- GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
- GPIO_Init(GPIOA, &GPIO_InitStructure);//根据设置配置SPI相关GPIO口
- GPIO_SetBits(GPIOA,GPIO_Pin_4|GPIO_Pin_5|GPIO_Pin_6|GPIO_Pin_7);
- SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;//设置SPI单向或者双向的数据模式:SPI设置为双线双向全双工
- SPI_InitStructure.SPI_Mode = SPI_Mode_Master;//设置SPI工作模式:设置为主SPI
- SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;//设置SPI的数据大小:SPI发送接收8位帧结构
- SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;//串行同步时钟的空闲状态为高电平
- SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;//串行同步时钟的第二个跳变沿(上升或下降)数据被采样
- SPI_InitStructure.SPI_NSS = SPI_NSS_Hard;//NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制
- SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_256;//定义波特率预分频的值:波特率预分频值为256
- SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;//指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始
- SPI_InitStructure.SPI_CRCPolynomial = 7;//CRC值计算的多项式
- SPI_Init(SPI1, &SPI_InitStructure);//根据SPI_InitStruct中指定的参数初始化外设SPIx寄存器
- SPI_SSOutputCmd(SPI1, ENABLE);//使能SS位输出使能(因为使用的是硬件NSS信号)
- SPI_Cmd(SPI1, ENABLE); //使能SPI外设
- }
- //SPI 速度设置函数
- //SpeedSet:
- //SPI_BaudRatePrescaler_2 2分频
- //SPI_BaudRatePrescaler_8 8分频
- //SPI_BaudRatePrescaler_16 16分频
- //SPI_BaudRatePrescaler_256 256分频
- void SPI1_SetSpeed(u8 SPI_BaudRatePrescaler)
- {
- assert_param(IS_SPI_BAUDRATE_PRESCALER(SPI_BaudRatePrescaler));
- SPI1->CR1&=0XFFC7;
- SPI1->CR1|=SPI_BaudRatePrescaler; //设置SPI2速度
- SPI_Cmd(SPI1,ENABLE);
- }
- //SPIx 读写一个字节
- //TxData:要写入的字节
- //返回值:读取到的字节
- u8 SPI1_ReadWriteByte(u8 TxData)
- {
- u8 retry=0;
- while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_TXE) == RESET) //检查指定的SPI标志位设置与否:发送缓存空标志位
- {
- retry++;
- if(retry>200)
- return 0;
- }
- SPI_I2S_SendData(SPI1, TxData); //通过外设SPIx发送一个数据
- retry=0;
- while (SPI_I2S_GetFlagStatus(SPI1, SPI_I2S_FLAG_RXNE) == RESET) //检查指定的SPI标志位设置与否:接受缓存非空标志位
- {
- retry++;
- if(retry>200)
- return 0;
- }
- return SPI_I2S_ReceiveData(SPI1); //返回通过SPIx最近接收的数据
- }
复制代码 DMA代码部分(使用的是DMA循环传输方式,全程无需CPU干预):
- #include "dma.h"
- /**
- * @brief SPI1使用DMA发送配置函数,内存到外设(SPI1->DR)
- * [url=home.php?mod=space&uid=271674]@param[/url] 无
- * @retval 无
- */
- void SPI_DMA_TxConfig(DMA_Channel_TypeDef* DMA_CHx,u32 cpar,u32 cmar,u16 cndtr)
- {
- /*DMA初始化结构体*/
- DMA_InitTypeDef DMA_InitStructure;
- /*使能所需时钟*/
- RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);//开启DMA时钟,注意DMA挂载在AHB总线上
- /*将DMA的通道1寄存器重设为缺省值*/
- DMA_DeInit(DMA_CHx);
- /*配置串口3DMA相关参数*/
- DMA_InitStructure.DMA_PeripheralBaseAddr = cpar;//设置DMA外设地址为串口1的DR寄存器地址
- DMA_InitStructure.DMA_MemoryBaseAddr = cmar;//设置内存地址(要传输变量的指针)为数组首地址
- DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;//配置传输方向:内存到外设
- DMA_InitStructure.DMA_BufferSize = cndtr;//配置要传输的数目
- DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;//开启内存地址递增
- DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//禁用外设地址递增
- DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;//配置外设数据单位为一个字节
- DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;//配置内存数据单位为一个字节
- DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;//配置DMA模式为循环
- DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;//配置请求的优先级为中等
- DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;//不开启内存到内存之间的传输
- /*初始化配置DMA*/
- DMA_Init(DMA_CHx, &DMA_InitStructure);//初始化使用到的DMA通道:SPI1的发送对应DMA1的通道3
- /*配置SPI1的DMA发送标志*/
- SPI1->CR2 |= 1<<1;
- }
复制代码 OLED.c代码(OLED初始化时要设置成自动地址+1并循环):
- #include "oled.h"
- #include "delay.h"
- #include "spi.h"
- #include "dma.h"
- /**
- * SPI时钟线:GPIOA5
- * SPI数据线:GPIOA7
- * SPI片选线:GPIOA4,使用硬件方式
- * OLED复位线:GPIOA3
- * OLED命令/数据选择线:GPIOA2
- */
- u8 DisplayBuff[8][128];
- void OLED_Init(void)
- {
- GPIO_InitTypeDef GPIO_InitStructure;
-
- RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
- /*初始化命令/数据控制引脚及复位引脚*/
- GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2|GPIO_Pin_3;//GPIOA2:命令数据选择、GPIOA3:复位引脚
- GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
- GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
- GPIO_Init(GPIOA, &GPIO_InitStructure);
- GPIO_SetBits(GPIOA,GPIO_Pin_2|GPIO_Pin_3);
-
- SPI1_Init();//初始化SPI1
- SPI1_SetSpeed(SPI_BaudRatePrescaler_2);//设置SPI1速度
- SPI_DMA_TxConfig(DMA1_Channel3,(u32)(&SPI1->DR),(u32)DisplayBuff,1024);
-
- OLED_Reset();
- OLED_WriteCMD(0xAE);//OLED休眠
- OLED_WriteCMD(0x00);//为页寻址模式设置下栏起始地址
- OLED_WriteCMD(0x10);//为页寻址模式设置更高的列起始地址
- OLED_WriteCMD(0x40);//设置显示起始线(0x00~0x3F)
- OLED_WriteCMD(0x81);//设置对比度控制寄存器
- OLED_WriteCMD(0xCF);//设置SEG输出电流亮度
- OLED_WriteCMD(0xA1);//Set SEG/Column Mapping 0xa0左右反置 0xa1正常
- OLED_WriteCMD(0xC8);//Set COM/Row Scan Direction 0xc0上下反置 0xc8正常
- OLED_WriteCMD(0xA8);//set multiplex ratio(1 to 64)
- OLED_WriteCMD(0x3f);//1/64 duty
- OLED_WriteCMD(0x81);//对比度设置
- OLED_WriteCMD(0x12);//1~255;默认0X7F (亮度设置,越大越亮)
- OLED_WriteCMD(0xD3);//设置显示偏移 (0x00~0x3F)
- OLED_WriteCMD(0x00);//不偏移
- OLED_WriteCMD(0xd5);//时钟频率
- OLED_WriteCMD(0x80);//100 Frames/Sec
- OLED_WriteCMD(0xD9);//预充电持续时间
- OLED_WriteCMD(0xF1);//Set Pre-Charge as 15 Clocks & Discharge as 1 Clock
- OLED_WriteCMD(0xDA);//设置com引脚硬件配置
- OLED_WriteCMD(0x12);//
- OLED_WriteCMD(0xDB);//set vcomh
- OLED_WriteCMD(0x40);//Set VCOM Deselect Level
- OLED_WriteCMD(0x20);//设置内存寻址模式
- OLED_WriteCMD(0x00);//设置成自动巡回扫描模式,先行后纵
- OLED_WriteCMD(0x8D);//设置电荷泵
- OLED_WriteCMD(0x14);//开启电荷泵
- OLED_WriteCMD(0xA4);//Disable Entire Display On (0xa4/0xa5)
- OLED_WriteCMD(0xA6);//正常显示模式,1为亮0为灭
- OLED_WriteCMD(0xAF);//OLED唤醒
- }
- /*写命令*/
- void OLED_WriteCMD(u8 data)
- {
- GPIO_ResetBits(GPIOA,GPIO_Pin_2);
- SPI1_ReadWriteByte(data);
- GPIO_SetBits(GPIOA,GPIO_Pin_2);
- }
- /*写数据*/
- void OLED_WriteDATA(u8 data)
- {
- GPIO_SetBits(GPIOA,GPIO_Pin_2);
- SPI1_ReadWriteByte(data);
- GPIO_ResetBits(GPIOA,GPIO_Pin_2);
- }
- /*复位*/
- void OLED_Reset(void)
- {
- GPIO_ResetBits(GPIOA,GPIO_Pin_3);
- delay_ms(100);
- GPIO_SetBits(GPIOA,GPIO_Pin_3);
- }
- /*重置下一比特位置到显示屏的起始位*/
- void OLED_Start(void)
- {
- OLED_WriteCMD (0xb0);//设置页地址(0~7)
- OLED_WriteCMD (0x00);//设置显示位置—列低地址
- OLED_WriteCMD (0x10);//设置显示位置—列高地址
- DMA_Cmd(DMA1_Channel3, ENABLE);//使能SPI1 TX DMA1 所指示的通道
- }
- /*开启OLED显示,还不完整,若是已经进入DMA传输,请先失能DMA传输再调用此函数*/
- void OLED_Display_On(void)
- {
- OLED_WriteCMD(0X8D);//SET DCDC命令
- OLED_WriteCMD(0X14);//DCDC ON
- OLED_WriteCMD(0XAF);//DISPLAY ON
- }
- /*关闭OLED显示,还不完整,若是已经进入DMA传输,请先失能DMA传输再调用此函数*/
- void OLED_Display_Off(void)
- {
- OLED_WriteCMD(0X8D);//SET DCDC命令
- OLED_WriteCMD(0X10);//DCDC OFF
- OLED_WriteCMD(0XAE);//DISPLAY OFF
- }
复制代码
|
|