系列文章目录
【STM32】| 01——常用外设 | USART
【STM32】| 02——常用外设 | I2C
失败了也挺可爱,成功了就超帅。 |
文章目录
- 前言
- 1. 简介
- 2. I2C协议
- 2.1 I2C物理连接
- 2.2 I2C通信协议
- 2.2.1 起始和停止信号
- 2.2.2 数据有效性
- 2.2.3 数据传输格式
- 2.2.4 从机地址/数据方向(R/W位)
- 2.2.5 响应与不响应(ACK/NACK)
- 2.2.6 同步与仲裁机制
- 1. SCL同步
- 2. SDA仲裁
- 2.3 I2C读写操作过程
- 2.3.1 Master向7位地址Savle写数据
- 2.3.2 Master从7位地址Savle读取数据
- 2.3.3 Master向7位地址Savle写/读取数据 (组合通信)
- 2.3.4 Master向/从10位地址Savle写/读取数据
- 3. MCU的I2C外设
- 4. I2C驱动函数
- 4.1 硬件I2C
- 4.1.1 轮询(阻塞)式
- 1、Cubemx配置
- 2、MCU作主机发送数据给从机
- 3、MCU作主机读取从机数据
- 4.1.2 中断式
- 1、Cubemx配置
- 2、MCU作主机发送数据给从机
- 3、MCU作主机读取从机数据
- 4.1.3 DMA
- 1、Cubemx配置
- 2、MCU作主机发送数据给从机
- 3、MCU作主机读取从机数据
- 4.1.4 HAL库I2C接口
- 4.2 IO模拟I2C
- 看查波形 验证IO模拟 发送一个字节
- 看查波形 验证IO模拟 应答和不应答
- 5. 使用I2C与从设备通信
- 5.1 驱动i2c接口的OLED屏幕
- 5.1.1 了解我们的屏幕
- 5.1.2 阅读SSD1306手册
- 1、获取MCU I2C与SSD2306交互的帧格式
- 2、看查命令表及详细介绍
- 3、显示原理及寻址模式
- 寻址模式
- 显示原理
- 取模
- 4、初始化流程
- 5.1.4 编写OLED驱动代码
- 硬件I2C
- 给OLED写一个字节命令/数据
- 给OLED写多个字节数据
- OLED清空
- OLED显示一个字符
- OLED显示字符串
- OLED显示数字
- OLED初始化
- 软件IO模拟I2C
- 给OLED写一个字节命令/数据
- 给OLED写多个字节数据
前言
本文详细介绍 I2C协议及 MCU I2C配置使用
1. 简介
I2C是一种常用的串行通信总线,由串行数据线SDA 和串线时钟线SCL组成。I2C是一种多主机控制总线,由飞利浦公司为了让主板、嵌入式系统或手机用以连接低速周边设备而发展。I2C 通讯协议(Inter-Integrated Circuit)是由于它引脚少,硬件实现简单,可扩展性强,不需要外部收发设备,被广泛地使用在系统内多个集成电路(IC)间的通讯。
I2C支持 0KHZ-5MHZ设备通信(hz相当于bps)。有如下几种模式
-
普通模式 ——100kHz
-
快速模式——400kHz
-
快速模式——1MHz
-
高速模式——3.4MHz
-
超高速模式——5MHz
我们常用400KHZ 在此基础上 Inter提出了SMBUS系统总线管理 该规范限制了通信速率10K-100KHZ.
I2C是一种主从通信 支持多主多从的总线。
2. I2C协议
2.1 I2C物理连接
如图 可以看到索引 I2C 设备都通过 SDA/SCL 连接到总线 总线接有上拉电阻(后面讲原因)
I2C特性如下:
- 总线只需两条线路:一条串行数据线 SDA 一条串行时钟线 SCL并利用电阻将电位上拉
- 每个连接到总线的器件都可以通过唯一的地址和一直存在的简单的主机/从机关系软件设定地址,主机可以作为主机发送器或主机接收器
- 它是一个真正的多主机总线 如果两个或更多主机同时初始化数据传输可以通过冲突检测和仲裁机制可防止数据被破坏
- 串行的 8位双向数据传输位速率在标准模式下可达 100kbit/s 快速模式下可达 400kbit/s 高速模式下可达 3.4Mbit/s 单向传输可以高达5Mbit/s
- 片上的滤波器可以滤去总线数据线上的毛刺波保证数据完整
- 连接到相同总线的 IC 数量只受到总线的最大电容 400pF 限制
SCL和SDA都是双向的通过上拉电阻连接电源 总线在空闲时都输出高电平 总线具有 线与功能。
2.2 I2C通信协议
该协议约定了通信的起始、停止信号以及数据有效性、响应、仲裁同步、地址广播等。
2.2.1 起始和停止信号
起始信号:SCL高电平时,SDA由高电平转换为低电平
停止信号:SCL高电平时,SDA由低电平转换为高电平
起始信号和停止信号一般都是主机发出,当有起始信号时,总线就会处于被占用状态,当有停止信号时,总线处于空闲状态。
2.2.2 数据有效性
SDA数据线在 SCL 的每个时钟周期传输一位数据。传输时, SCL 为高电平的时候 SDA 表示的数据有效,即此时的 SDA 为高电平时表示数据“1”,为低电平时表示数据“0”。当 SCL为低电平时, SDA 的数据无效,一般在这个时候 SDA 进行电平切换(数据位切换高/低),为下一位要传输的数据做好准备(即下一位要传1 SDA切换为高电平反之低平)。
2.2.3 数据传输格式
SDA 线上的每个字节必须为 8 位 每次传输可以发送的字节数量不受限制 每个字节后必须跟一个响应位 首先传输的是数据的最高位 MSB 如果从机要完成一些其他功能后(例如一个内部中断服务程序) 才能接收或发送下一个完整的数据字节 可以使时钟线 SCL 保持低电平迫使主机进入等待状态 当从机准备好接收下一个数据字节并释放时钟线 SCL 后 数据传输继续。
2.2.4 从机地址/数据方向(R/W位)
MSB:代表高位
如图所示,为一个7位地址,I2C 总线上的每个设备都有自己的独立地址,主机发起通讯时,通过 SDA 信号线发送设备地址(SLAVE_ADDRESS)来查找从机。 I2C 协议规定设备地址可以是 7 位或 10 位。
设备地址后面的一个数据位R/W用来表示数据传输方向,数据方向位为“1”时表示主机由从机读数据,该位为“0”时表示主机向从机写数据
下图是完整的传输时序
2.2.5 响应与不响应(ACK/NACK)
I2C的数据和地址传输都带响应。响应包括“应答(ACK)”和“非应答(NACK)”两种信号。
数据传输时主机产生时钟,在第 9 个时钟时,数据发送端会释放 SDA 的控制权,由数据接收端控制 SDA,若 SDA 为高电平,表示非应答信号(NACK),低电平表示应答信号(ACK)。
作为数据接收端时,当设备(主/从机)接收到 I2C 传输的一个字节数据或地址后,
若希望对方继续发送数据,则需要向对方发送“应答(ACK)”信号,发送方会继续发送下
一个数据;若接收端希望结束数据传输,则向对方发送“非应答(NACK)”信号,发送方接
收到该信号后会产生一个停止信号,结束信号传输。
2.2.6 同步与仲裁机制
总线上的设备可以抽象为节点。在多主通信中,总线上会有很多节点,它们都有自己的寻址地址,可以作为从节点被别的节点访问,同时它们都可以作为主节点向其他的节点发送控制字节和传送数据。但是如果有两个或两个以上的节点都向总线上发送请求时,这样就形成了冲突。要解决这种冲突,就要进行同步/仲裁,这就是I 2C总线上的同步/仲裁。
同步指:SCL同步
仲裁指:SDK仲裁
1. SCL同步
SCL同步是由于总线具有线“与”的逻辑功能。
1、只要有一个节点发送低电平时,总线上就表现为低电平。
2、当所有节点都发送高电平时,总线上就表现为高电平。
2. SDA仲裁
SDA线的仲裁也是建立在总线具有线“与”逻辑功能上的。节点在发送1位数据后,比较SDA线上所呈现的数据与自己发送的是否一致。是,继续发送;否则,退出竞争。SDA线的仲裁可以保证I2C总线系统在多个主节点同时企图控制总线时通信正常进行并且数据不丢失。总线系统通过仲裁只允许一个主节点可以继续占据总线。
上图是以两个节点为例的仲裁过程。DATA1和DATA2分别是主节点向总线所发送的数据信号,SDA为总线上所呈现的数据信号,SCL是总线上所呈现的时钟信号。
当主节点1、2同时发送起始信号时,两个主节点都发送了高电平信号。这时总线上呈现的信号为高电平,两个主节点都检测到总线上的信号与自己发送的信号相同,继续发送数据。
第2个时钟周期,2个主节点都发送低电平信号,在总线上呈现的信号为低电平,仍继续发送数据。
在第3个时钟周期,主节点1发送高电平信号,而主节点2发送低电平信号。根据总线的线“与”的逻辑功能,总线上的信号为低电平,这时主节点1检测到总线上的数据和自己所发送的数据不一样,就断开数据的输出级,转为从机接收状态。这样主节点2就赢得了总线,而且数据没有丢失,即总线的数据与主节点2所发送的数据一样,而主节点1在转为从节点后继续接收数据,同样也没有丢掉SDA线上的数据。因此在仲裁过程中数据没有丢失。
总结:SDA仲裁和SCL时钟同步处理过程没有先后关系,而是同时进行的。
2.3 I2C读写操作过程
下面通过主机到从机的读写操作进行介绍
主机产生起始信号后,所有从机就开始等待主机紧接下来 广播 的从机地址信号
(SLAVE_ADDRESS)。 在 I2C 总线上,每个设备的地址都是唯一的, 当主机广播的地址与
某个设备地址相同时,这个设备就被选中了,没被选中的设备将不回接受之后的数据信号。
根据 I2C 协议,从机地址可以是 7 位或 10 位。
在地址位之后,是(RW位)传输方向的选择位,该位为 0 时,表示后面的数据传输方向是由主机传输至从机,即主机向从机写数据。该位为 1 时,则相反,即主机由从机读数据。
从机接收到匹配的地址后,主机或从机会返回一个应答(ACK)或非应答(NACK)信号,
只有接收到应答信号后,主机才能继续发送或接收数据。
2.3.1 Master向7位地址Savle写数据
斜线:主机传输至从机 S :传输开始信号 SLAVE_ADDRESS: 从机地址
空白:从机传输至主机 A/A:应答(ACK)或非应答(NACK)信号
I2C主机设备 向一个具有7位地址的I2C从机设备写入N个字节数据的数据帧格式:主机先发送开始信号+7位地址+1位R/W位+响应位(从机响应ACK)继续传输+[N字节数据+从机ACK]
如果从机不想接收了 回应NACK 停止传输 主机发送停止信号
2.3.2 Master从7位地址Savle读取数据
斜线:主机传输至从机 S :传输开始信号 SLAVE_ADDRESS: 从机地址
空白:从机传输至主机 A/A:应答(ACK)或非应答(NACK)信号 DATA数据
若数据方向位配置为“1读数据”方向, 即如图所示, 主机发送起始位+广播完地址,等待接收到从机应答信号后, 从机开始向主机返回数据(DATA),数据包大小也为 8 位,从机每发送完一个数据,都会等待主机的应答信号(ACK),重复这个过程,可以返回 N 个数据,这个 N 也没有大小限制。当主机希望停止接收数据时,就向从机返回一个非应答信号(NACK),则从机自动停止数据传输。
2.3.3 Master向7位地址Savle写/读取数据 (组合通信)
除了单独的读和写, I2C 通讯更常用的是复合格式,即如图所示,
该传输过程有两次起始信号(S)。一般在第一次传输中,主机通过 SLAVE_ADDRESS 寻找到从设备后,发送一段“数据”,这段数据通常用于表示从设备内部的寄存器或存储器地址(注意区分它与SLAVE_ADDRESS 的区别);在第二次的传输中,对该地址的内容进行读或写。也就是说,第一次通讯是告诉从机,主机要读写从机的地址,第二次则是读写的实际内容。
2.3.4 Master向/从10位地址Savle写/读取数据
10为地址用作扩展 大多数都是7位地址 暂时不详细说啦 后面遇到了在填充
Master向10位地址Savle写数据过程
TODO
3. MCU的I2C外设
各种信号MCU大差不差 这里以stm32说明
stm32 i2c外设它提供多主机功能,控制所有I2C总线特定的时序、协议、仲裁和定时。支持标准和快速两种模式,与SMBus 2.0兼容。具备状态错误检测标志及中断、可DMA等特点
详细功能、寄存器等描述 看参考手册
I2C功能框图
四种模式
● 从发送器模式
● 从接收器模式
● 主发送器模式
● 主接收器模式
中断事件类型
4. I2C驱动函数
一般我们i2c与其他设备通信 所以 I2C是一个工具 配置好I2C 发送/接收后 还需要根据从设备做一些具体操作
旧============================================~~
我这里有个 I2C接口的 温湿度传感器 AHT10 这里用这个演示
先配置单片机 I2C 读写
修改===========================================
这里用OLED来演示 I2C通信 AHT10不知道什么原因 发送设备地址不响应
4.1 硬件I2C
硬件I2C指 MCU自带的I2C外设 有固定引脚
HAL库提供了很多接口 作为主机的收发 作为从机的收发 存储设备读写 以及对应的3种方式(阻塞、中断、DMA) 这里不全部介绍了 只说下用到的
4.1.1 轮询(阻塞)式
1、Cubemx配置
I2C1 默认功能引脚是 PB6/7 它会自动配置我们不用管
从设备地址 根据从设备数据手册 看查 我这里AHT10传感器的 设备地址为 0X38
2、MCU作主机发送数据给从机
因为我们要让OLED显示我们想要的内容 所以要给OLED写命令/数据给它 OLED如何配置这些最后在说 暂时只演示 收发接口
这里我们用主机的收发API
HAL_StatusTypeDef HAL_I2C_Master_Transmit(I2C_HandleTypeDef *hi2c,uint16_t DevAddress,uint8_t *pData,uint16_t Size,uint32_t Timeout);
可以看到信号正常发出去了 只是没有应答 如果设备地址正确 从设备会响应
设备地址正确 从设备响应 就可以通信了 可以看到写的数据也可以写进去了
3、MCU作主机读取从机数据
HAL_StatusTypeDef HAL_I2C_Master_Receive(I2C_HandleTypeDef *hi2c,uint16_t DevAddress,uint8_t *pData,uint16_t Size,uint32_t Timeout);
4.1.2 中断式
1、Cubemx配置
2、MCU作主机发送数据给从机
和阻塞使用方法一样 只是调中断的接口
HAL_StatusTypeDef HAL_I2C_Master_Transmit_IT(I2C_HandleTypeDef *hi2c,uint16_t DevAddress,uint8_t *pData,uint16_t Size,uint32_t Timeout)
如果传输完成调用发送完成回调函数
void HAL_I2C_MasterTxCpltCallback(I2C_HandleTypeDef *hi2c)
{/* 如果传输完成会进入这里 */if(hi2c1.Instance==I2C1){}
}
3、MCU作主机读取从机数据
HAL_StatusTypeDef HAL_I2C_Master_Receive_IT(I2C_HandleTypeDef *hi2c,uint16_t DevAddress,uint8_t *pData,uint16_t Size,uint32_t Timeout);
如果接收完成调用接收完成回调函数
void HAL_I2C_MasterRxCpltCallback(I2C_HandleTypeDef *hi2c)
{/* 如果接收完成会进入这里 */if(hi2c1.Instance==I2C1){}
}
4.1.3 DMA
1、Cubemx配置
2、MCU作主机发送数据给从机
和阻塞使用方法一样 只是调中断的接口
HAL_StatusTypeDef HAL_I2C_Master_Transmit_DMA(I2C_HandleTypeDef *hi2c,uint16_t DevAddress,uint8_t *pData,uint16_t Size,uint32_t Timeout)
如果传输完成调用发送完成回调函数
void HAL_I2C_MasterTxCpltCallback(I2C_HandleTypeDef *hi2c)
{/* 如果传输完成会进入这里 */if(hi2c1.Instance==I2C1){}
}
3、MCU作主机读取从机数据
HAL_StatusTypeDef HAL_I2C_Master_Receive_DMA(I2C_HandleTypeDef *hi2c,uint16_t DevAddress,uint8_t *pData,uint16_t Size,uint32_t Timeout);
如果接收完成调用接收完成回调函数
void HAL_I2C_MasterRxCpltCallback(I2C_HandleTypeDef *hi2c)
{/* 如果接收完成会进入这里 */if(hi2c1.Instance==I2C1){}
}
4.1.4 HAL库I2C接口
接口很多大差不差
HAL_StatusTypeDef HAL_I2C_Master_Transmit(I2C_HandleTypeDef *hi2c,uint16_t DevAddress,uint8_t *pData,uint16_t Size,uint32_t Timeout);// 一般用作写EEPROM 存储设备方便
HAL_StatusTypeDef HAL_I2C_Mem_Write(I2C_HandleTypeDef*hi2c, uint16_t DevAddress, uint16_t MemAddress,// 相当于从设备寄存器地址/命令uint16_t MemAddSize,uint8_t *pData, // 要写的数据uint16_t Size, uint32_t Timeout)#define dev_addr 0x78/* 第一个数据 比如从机的某个寄存器/命令 *//* 第2/3个数据 要写入的数据 */uint8_t data[3]={0x31,0x10,0x11};/* 以下两个等价 */HAL_I2C_Master_Transmit(&hi2c1,dev_addr,data,3,0xffff);HAL_I2C_Mem_Write(&hi2c1,dev_addr,data[0],1,&data[1],2,0xffff);
4.2 IO模拟I2C
使用IO口去模拟 I2C 不需要固定引脚 灵活
根据I2C协议 实现I2C读写功能
/* 1、定义引脚并配置IO模式 */
#define I2C_WR 0 /* 写控制bit */
#define I2C_RD 1 /* 读控制bit */#define I2C_GPIO_CLK_ENABLE() __HAL_RCC_GPIOB_CLK_ENABLE()
#define I2C_GPIO_PORT GPIOB
#define I2C_SCL_PIN GPIO_PIN_6
#define I2C_SDA_PIN GPIO_PIN_7#define I2C_SCL_HIGH() HAL_GPIO_WritePin(I2C_GPIO_PORT,I2C_SCL_PIN,GPIO_PIN_SET) // 输出高电平
#define I2C_SCL_LOW() HAL_GPIO_WritePin(I2C_GPIO_PORT,I2C_SCL_PIN,GPIO_PIN_RESET) // 输出低电平
#define I2C_SDA_HIGH() HAL_GPIO_WritePin(I2C_GPIO_PORT,I2C_SDA_PIN,GPIO_PIN_SET) // 输出高电平
#define I2C_SDA_LOW() HAL_GPIO_WritePin(I2C_GPIO_PORT,I2C_SDA_PIN,GPIO_PIN_RESET) // 输出低电平
#define I2C_SDA_READ() HAL_GPIO_ReadPin(I2C_GPIO_PORT,I2C_SDA_PIN)void sw_i2c_init(void)
{GPIO_InitTypeDef GPIO_InitStruct;/* 打开GPIO时钟 */I2C_GPIO_CLK_ENABLE();GPIO_InitStruct.Pin = I2C_SCL_PIN|I2C_SDA_PIN;GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_OD;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_MEDIUM;HAL_GPIO_Init(I2C_GPIO_PORT, &GPIO_InitStruct);
}
/* 2、I2C起始信号/停止信号 */
static void I2C_Delay(void)
{uint8_t i;for (i = 0; i < 10; i++);
}
void I2C_Start(void)
{/* SCL高电平时 SDA由高变低 */I2C_SDA_HIGH();I2C_SCL_HIGH();I2C_Delay();I2C_SDA_LOW();I2C_Delay();I2C_SCL_LOW();I2C_Delay();
}
void I2C_Stop(void)
{/* SCL高电平时,SDA由低变高 */I2C_SDA_LOW();I2C_SCL_HIGH();I2C_Delay();I2C_SDA_HIGH();
}
/* 3、I2C应答信号/不应答/等待检测应答信号 */
void I2C_Ack(void)
{I2C_SDA_LOW(); /* CPU驱动SDA = 0 */I2C_Delay();I2C_SCL_HIGH(); /* CPU产生1个时钟 */I2C_Delay();I2C_SCL_LOW();I2C_Delay();I2C_SDA_HIGH(); /* CPU释放SDA总线 */
}
void I2C_NAck(void)
{I2C_SDA_HIGH(); /* CPU驱动SDA = 1 */I2C_Delay();I2C_SCL_HIGH(); /* CPU产生1个时钟 */I2C_Delay();I2C_SCL_LOW();I2C_Delay();
}
uint8_t I2C_WaitAck(void)
{uint8_t re;I2C_SDA_HIGH(); /* CPU释放SDA总线 */I2C_Delay();I2C_SCL_HIGH(); /* CPU驱动SCL = 1, 此时器件会返回ACK应答 */I2C_Delay();if (I2C_SDA_READ()) /* CPU读取SDA口线状态 */{re = 1;}else{re = 0;}I2C_SCL_LOW();I2C_Delay();return re;
}
/* 4、写/读一个字节数据 */
void I2C_Write_One_Byte(uint8_t Byte)
{uint8_t i;/* 先发送字节的高位bit7 */for (i = 0; i < 8; i++){ if (Byte & 0x80){I2C_SDA_HIGH();}else{I2C_SDA_LOW();}I2C_Delay();I2C_SCL_HIGH();I2C_Delay(); I2C_SCL_LOW();if (i == 7){I2C_SDA_HIGH(); // 释放总线}Byte <<= 1; /* 左移一个bit */I2C_Delay();}
}
uint8_t I2C_Read_One_Byte(void)
{uint8_t i;uint8_t value;/* 读到第1个bit为数据的bit7 */value = 0;for (i = 0; i < 8; i++){value <<= 1;I2C_SCL_HIGH();I2C_Delay();if (I2C_SDA_READ()){value++;}I2C_SCL_LOW();I2C_Delay();}return value;
}
看查波形 验证IO模拟 发送一个字节
I2C_Start();I2C_Write_One_Byte(dev_addr);while (I2C_WaitAck() == 0) break;I2C_Stop();
可以看到是没问题 SCL时钟频率 222KHZ 这个和我们 Delay延时函数有关 我们用的软件 延时 变量=10 空循环10次 在系统时钟72Mhz下 频率 222KHZ 测试得i=29 100khz左右 i=3 400K左右
看查波形 验证IO模拟 应答和不应答
I2C_Start();I2C_Write_One_Byte(dev_addr);//I2C_NAck();I2C_Ack();I2C_Stop();
5. 使用I2C与从设备通信
以上介绍了 硬件I2C / 软件IO模拟I2C 驱动函数 后面我们讲解如何使用I2C 去和从设备通信
5.1 驱动i2c接口的OLED屏幕
5.1.1 了解我们的屏幕
我们了解3个信息 驱动芯片是那个 用什么方式驱动 屏幕分辨率
购买的时候 商品名称后缀会有 SSD1306等等 这指这个屏幕的 驱动芯片 这是模块内置的 我要用这个屏幕显示内容 就得去控制SSD1306这个OLED驱动芯片 给他什么命令还是什么内容 它才会让屏幕显示。
一般OLED屏幕有 I2C、SPI两种接口 如果我买了I2C的话 那就通过MCU I2C 和驱动SSD1306 SPI就用SPI
一般商家也会准备资料 资料里包含测试代码 屏幕产品手册 SSD1306驱动芯片手册 没的话就百度找下 不管驱动什么模块 用什么 看它的数据手册 是最好的资料
这是我手里的 0.96的OLED I2C的
从卖家给模块资料 可以了解到 我们屏幕 128x64个像素点
它的原理图
没什么东西 就是我们通过I2C去控制 SSD1306 如何驱动就是看它手册
5.1.2 阅读SSD1306手册
打开一看 60多页 还是英文 这看起来有点空难欸 不需要全看 找对我们有用的 那些有用呢
打开目录
1、获取MCU I2C与SSD2306交互的帧格式
我们看 MCU I2C接口章节 除了一些介绍 I2C相关的 发现一个最有用的 通信帧格式
2、看查命令表及详细介绍
第九章命令表 第10章命令详细描述
这些内容可以先不详细看 后面用到再查 我这里就梳理下用到的几个
有五类命令 :基本命令、滚动显示、显示地址设置、屏幕硬件配置、显示时钟频率
看到了D/C位的定义 这样的话
控制字节为 0X40表示写数据 0X00表示写命令
3、显示原理及寻址模式
刚看命令表 发现几个不懂的概念COM、SEG、 页地址、页寻址等 这时候我们在翻翻手册看看
寻址模式
第10章 设置寻址模式命令的详细描述 给出了解释
三种寻址模式:按页、水平、垂直
显示原理
128x64分辨率 放大了就是格子 我们想让OLED显示扫描内容 只需要按照要显示的内容
取模
通过取模软件 生成我们想要的数据
我演示一下
00H 00H 10H 10H F8H 00H 00H 00H 00H 00H 20H 20H 3FH 20H 20H 00H;“1”
这只是一个字符当我们要用常用字符呢 就需要生成字库了 汉字的字库太大
一般我们用ACSII英文字符的字库 和 用到那几个汉字 取模那几个汉字 后面会用到
4、初始化流程
看到一个 初始化流程及示例代码 具体操作查询下命令看看什么意思
这个官方SSD1306手册上的
下面这个是模块手册上的
5.1.4 编写OLED驱动代码
硬件I2C
先定义OLED.H
#ifndef __OLED_H
#define __OLED_H#ifdef __cplusplus
extern "C" {
#endif/* Includes ------------------------------------------------------------------*/
#include "stm32f1xx_hal.h"#define OLED_DEVICE_ADDR 0X78
/* OLED控制字节 */
#define OLED_CMD 0X00
#define OLED_DATA 0X40
/* 命令好多我这里就不详细定义了 不定义也可以用的时候查一下什么意思就行 */
/* 基本命令 ------------------------------------------------------------------*/
#define Set_Contrast_Control 0X81 /* 设置对比度 范围1-256 */
#define Entire_Display_ON 0xA4 /* 全局显示关 */
#define Entire_Display_OFF 0xA5 /* 全局显示开 */
#define Set_Normal_Display 0xA6 /* 正常显示 */
#define Set_Inverse_Display 0xA7 /* 反向显示 */
#define Set_Display_ON 0xAE /* 显示开 */
#define Set_Display_OFF 0XAF /* 显示关 */
/* 地址设置命令 --------------------------------------------------------------*/
#define Set_Lower_Column(x) ((x<=0x0F)?x:0) /* 设置列的起始地址低位 范围0x00-0x0F */
#define Set_Higher_Column(x) ((x>=0x10&&x<=0x1F)?x:0x10) /* 设置列的起始地址高位 范围0x10-0x1F */
#define Set_Page_Start_Address(x) ((x>=0xB0&&x<=0xB7)?x:0XB0) /* 设置显示页起始地址 *//*** [oled_init OLED初始化]*/
void oled_init(void);
/*** [oled_show_string OLED显示字符串]* @param x [X坐标]* @param y [y坐标]* @param str [字符串]*/
void oled_show_char(uint8_t x,uint8_t y,uint8_t ch);
/*** [oled_show_string OLED显示字符串]* @param x [X坐标]* @param y [y坐标]* @param ch [ch]*/
void oled_show_string(uint8_t x,uint8_t y,uint8_t *str);
/*** [oled_show_num OLED显示数字]* @param x [X坐标]* @param y [X坐标]* @param num [显示的数]*/
void oled_show_num(uint8_t x,uint8_t y,uint32_t num);#ifdef __cplusplus
}
#endif#endif
OLED.c
好多内容没保存 想哭死呜呜呜 我闲的点关机呜呜呜 哎明天再搞以下了呜呜呜
给OLED写一个字节命令/数据
void oled_write_one_byte(uint8_t ctl, uint8_t data)
{HAL_I2C_Mem_Write(&hi2c1, OLED_DEVICE_ADDR, ctl, 1, &data, 1, 0xff);
#if 0uint8_t temp[2] = {0};temp[0] = ctl;temp[1] = data;HAL_I2C_Master_Transmit(&hi2c1, OLED_DEVICE_ADDR, temp, 2, 0xff);
#endif
}
给OLED写多个字节数据
void oled_write_byte(uint8_t* data, uint8_t size)
{HAL_I2C_Mem_Write(&hi2c1, OLED_DEVICE_ADDR, OLED_DATA, 1, data, size, 0xff); }
OLED清空
uint8_t OLED_GRAM[128][16];
void oled_clear(void)
{uint8_t i, n;for(i = 0; i < 8; i++) {for(n = 0; n < 128; n++) {/* 清空数据 */OLED_GRAM[n][i] = 0;}}/* 更新显存 */for(i = 0; i < 8; i++) {/* 设置行起始地址 */oled_write_one_byte(OLED_CMD, 0xb0 + i);/* 设置列起始地址低位 */oled_write_one_byte(OLED_CMD, 0x00);/* 设置列起始地址高位 */oled_write_one_byte(OLED_CMD, 0x10);/* 按行写数据到显存 */oled_write_byte(&OLED_GRAM[0][0], 128);}
}
OLED显示一个字符
void oled_show_char(uint8_t x,uint8_t y,uint8_t ch)
{uint8_t page = y;uint8_t col = 8;if(y > 7 || x > 15)return;oled_write_one_byte(OLED_CMD, 0xB0 + page&0x0f);oled_write_one_byte(OLED_CMD,0x00+col&0x0f);oled_write_one_byte(OLED_CMD, 0x10+col>>4);oled_write_byte((uint8_t*)&ascii_font_8x16[ch][0], 8);oled_set_pos(page + 1, col);oled_write_byte((uint8_t*)&ascii_font_8x16[ch][8], 8);
}
OLED显示字符串
void oled_show_string(uint8_t x,uint8_t y,uint8_t *str)
{
uint8_t i=0;while(str[i]){oled_put_char(x, y, str[i]);x++;if(x > 15) {x = 0;y += 2;}i++;}
}
OLED显示数字
void oled_show_num(uint8_t x,uint8_t y,uint32_t num)
{uint8_t str[16]={0};sprintf((char *)&str,"%d",num);oled_put_string(x, y, str);
}
OLED初始化
void oled_init(void)
{/* 1、初始化I2C */MX_I2C1_Init();/* 2、配置OLED *//* OLED Demo里 */oled_write_one_byte(OLED_CMD, 0xAE); //--turn off oled paneloled_write_one_byte(OLED_CMD, 0x00); //---set low column addressoled_write_one_byte(OLED_CMD, 0x10); //---set high column addressoled_write_one_byte(OLED_CMD, 0x40); //--set start line address Set Mapping RAM Display Start Line (0x00~0x3F)oled_write_one_byte(OLED_CMD, 0x81); //--set contrast control registeroled_write_one_byte(OLED_CMD, 0xCF); // Set SEG Output Current Brightnessoled_write_one_byte(OLED_CMD, 0xA1); //--Set SEG/Column Mapping 0xa0左右反置 0xa1正常oled_write_one_byte(OLED_CMD, 0xC8); // Set COM/Row Scan Direction 0xc0上下反置 0xc8正常oled_write_one_byte(OLED_CMD, 0xA6); // 正常模式oled_write_one_byte(OLED_CMD, 0xA8); //--set multiplex ratio(1 to 64)oled_write_one_byte(OLED_CMD, 0x3f); //--1/64 dutyoled_write_one_byte(OLED_CMD, 0xD3); //-set display offset Shift Mapping RAM Counter (0x00~0x3F)oled_write_one_byte(OLED_CMD, 0x00); //-not offsetoled_write_one_byte(OLED_CMD, 0xd5); //--set display clock divide ratio/oscillator frequencyoled_write_one_byte(OLED_CMD, 0x80); //--set divide ratio, Set Clock as 100 Frames/Secoled_write_one_byte(OLED_CMD, 0xD9); //--set pre-charge periodoled_write_one_byte(OLED_CMD, 0xF1); //Set Pre-Charge as 15 Clocks & Discharge as 1 Clockoled_write_one_byte(OLED_CMD, 0xDA); //--set com pins hardware configurationoled_write_one_byte(OLED_CMD, 0x12);oled_write_one_byte(OLED_CMD, 0xDB); //--set vcomholed_write_one_byte(OLED_CMD, 0x40); // Set VCOM Deselect Leveloled_write_one_byte(OLED_CMD, 0x20); //-Set Page Addressing Mode (0x00/0x01/0x02)oled_write_one_byte(OLED_CMD, 0x02); //oled_write_one_byte(OLED_CMD, 0x8D); //--set Charge Pump enable/disableoled_write_one_byte(OLED_CMD, 0x14); //--set(0x10) disableoled_write_one_byte(OLED_CMD, 0xA4); // Disable Entire Display On (0xa4/0xa5)oled_write_one_byte(OLED_CMD, 0xA6); // Disable Inverse Display On (0xa6/a7)oled_write_one_byte(OLED_CMD, 0xAF);oled_write_one_byte(OLED_CMD, 0xA6); //正常显示oled_write_one_byte(OLED_CMD, 0xC8); //正常显示oled_write_one_byte(OLED_CMD, 0xA1); //正常显示oled_clear();
}
软件IO模拟I2C
只需要替换 前面 硬件I2C初始化、读写为软件I2C就行
给OLED写一个字节命令/数据
void oled_write_one_byte(uint8_t ctl, uint8_t data)
{I2C_Start();I2C_Write_One_Byte(OLED_DEVICE_ADDR);I2C_WaitAck();if(ctl == OLED_DATA) {I2C_Write_One_Byte(OLED_DATA);} else {I2C_Write_One_Byte(OLED_CMD);}I2C_WaitAck();I2C_Write_One_Byte(data);I2C_WaitAck();I2C_Stop();
}
给OLED写多个字节数据
void oled_write_byte(uint8_t* data, uint8_t size)
{uint8_t i = 0;I2C_Start();I2C_Write_One_Byte(OLED_DEVICE_ADDR);I2C_WaitAck();I2C_Write_One_Byte(OLED_DATA);I2C_WaitAck();while(i < size) {I2C_Write_One_Byte(*data);I2C_WaitAck();data++;i++;}I2C_Stop();
}
替换这两个接口就好