【STM32】| 02——常用外设 | I2C

系列文章目录
【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();
}

替换这两个接口就好

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/414152.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

腾讯云轻量化应用服务器_轻量化应用服务器_轻量化私有云

腾讯云轻量应用服务器开箱即用、运维简单的轻量级云服务器&#xff0c;CPU内存带宽配置高并且价格特别便宜&#xff0c;大带宽&#xff0c;但是限制月流量&#xff0c;轻量2核2G3M带宽62元一年、2核2G4M优惠价118元一年&#xff0c;540元三年、2核4G5M带宽218元一年&#xff0c…

esp32-idf eclipse 定时器的使用demo

esp32定时器的使用demo 1、介绍 ESP32芯片包含两个硬件定时器组。每组有两个通用硬件定时器。它们都是基于16位预分频器和64位自动重载功能的向上向下计数器的64位通用定时器。 2、API接口函数 创建定时器函数&#xff1a; esp_timer_create(); esp_err_t esp_timer_create …

阿里云ECS使用docker搭建mysql服务

目录 1.确保正确安装好docker 2.安装mysql镜像 3.创建容器&#xff08;设置端口映射、目录映射&#xff09; 1.确保正确安装好docker 安装教程&#xff1a; 阿里云ECS(CentOS镜像)安装docker-CSDN博客https://blog.csdn.net/qq_62262918/article/details/135686614?spm10…

DAY12--learning English

一、积累 1.superstition 2.intersection If you are in an intersection and heard siren of an emergency vehicle. you should. 当你身处于道路交界处并且听到应急车辆的鸣笛声&#xff0c;你应该. 3. luxurious Japans $3000 Most luxurious sleeper train. 花费3000美元…

小程序商城 免 费 搭 建之java商城 电子商务Spring Cloud+Spring Boot+二次开发+mybatis+MQ+VR全景+b2b2c

java SpringCloud版本b2b2c鸿鹄云商平台全套解决方案 使用技术&#xff1a; Spring CloudSpring BootMybatis微服务服务监控可视化运营 B2B2C平台&#xff1a; 平台管理端(包含自营) 商家平台端(多商户入驻) PC买家端、手机wap/公众号买家端 微服务&#xff08;30个通用…

geemap学习笔记051:获取影像的范围

前言 通常在筛选的过程中&#xff0c;需要获取得到影像的边界范围&#xff0c;下面就将介绍一下如何获取影像的范围。 1 导入库并加载数据 import ee import geemapee.Initialize() # Create a map centered at (lat, lon). Map geemap.Map(center[40, -100], zoom4)image …

STM32 CubeIDE 使用 CMSIS-DAP烧录 (方法2--外部小工具)

前言&#xff1a; 本篇所用方法&#xff0c;需要借助一个外部的工具小软件。 优点&#xff1a;烧录更稳定; 缺点&#xff1a;不能在线仿真调试。 下面链接&#xff0c;是另一种方法&#xff1a;修改CubeIDE调试文件。能在CubeIDE直接烧录、仿真&#xff0c;但不稳定。…

The Blocks Problem

本题是一道模拟题&#xff0c;但个人感觉挺有意思的&#xff08;思路很明确&#xff0c;但是WA了好几发才过&#xff09;&#xff0c;因此来讲一讲思路。 题面 题面PDF 样例输入 10 move 9 onto 1 move 8 over 1 move 7 over 1 move 6 over 1 pile 8 over 6 pile 8 over 5…

ChatGPT与文心一言:AI助手之巅的对决

随着科技的飞速发展&#xff0c;人工智能助手已经渗透到我们的日常生活和工作中。 而在这个充满竞争的领域里&#xff0c;ChatGPT和文心一言无疑是最引人注目的两款产品。它们各自拥有独特的优势&#xff0c;但在智能回复、语言准确性、知识库丰富度等方面却存在差异。那么&am…

【新书推荐】Web3.0应用开发实战(从Web 2.0到Web 3.0)

第一部分 Flask简介 第1章 安装 1.1 创建应用目录 1.2 虚拟环境 1.2.1 创建虚拟环境 1.2.2 使用虚拟环境 1.3 使用pip安装Python包 1.4 使用pipregs输出包 1.5 使用requirements.txt 1.6 使用pipenv管理包 第2章 应用的基本结构 2.1 网页显示过程 2.2 初始化 2.3 路由和视图函数…

保姆版Vps安装灯塔(ARL)

因为灯塔的默认端口为5003&#xff0c;所以我们在安装之前就在防火墙里把我们的5003端口打开 打开端口步骤如下&#xff1a; 1.我们打开控制面板&#xff0c;在控制面板里点击 系统和安全 。如下图&#xff1a; 2.接着点击 Windows Defender防火墙,如下图&#xff1a; 3.再…

spring boot学习第八篇:kafka监听消费

为了实现监听器功能 pom.xml文件内容如下&#xff1a; <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0" xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLoc…