1. 初识0.96寸OLED:嵌入式开发的“迷你高清屏”
大家好,今天我想聊聊一个在嵌入式领域特别受欢迎的小玩意儿——0.96寸OLED显示屏。如果你玩过单片机,尤其是STM32或者Arduino,大概率见过这个只有拇指大小的屏幕。别看它尺寸小,能耐可不小:128x64的分辨率、自发光特性、超高对比度,还能显示中文和图像,简直是调试和项目展示的神器。
我第一次用这个屏幕是因为当时在做一个小型气象站,需要实时显示温湿度和气压数据。之前试过串口打印,但毕竟要连着电脑,不方便;也考虑过LCD屏,但接线多、功耗大,最后选了这款OLED。实测下来,它只需要4根线(I2C接口)就能驱动,功耗极低,而且显示效果非常清晰,阳光下也能看清。
这块屏的核心是SSD1306驱动芯片,它内部自带显存(GDDRAM)和控制器,支持多种寻址模式。我们通过I2C或SPI接口发送命令和数据,就能控制每个像素的亮灭。这种直接控制像素的方式,让我们能实现字符、图形甚至动画的显示,灵活性非常高。
对于初学者来说,0.96寸OLED是一个非常友好的入门选择。它价格便宜(某宝上10元左右),资料丰富,而且容易上手。不管是用来调试代码、显示传感器数据,还是做个小游戏机,都能胜任。接下来,我会带你从底层驱动到实战应用,彻底玩转这块小屏幕。
2. 深入核心:SSD1306驱动芯片与GDDRAM寻址原理
2.1 SSD1306芯片架构解析
SSD1306是这款OLED屏的核心大脑,它是一个单芯片CMOS OLED驱动器,专为共阴极OLED面板设计。芯片内部集成了对比度控制器、显示RAM(GDDRAM)和振荡器,大大减少了外部元件数量。这意味着我们只需要通过简单的通信协议,就能控制复杂的显示内容。
芯片的引脚中,关键的有:
- SCL和SDA:I2C通信的时钟线和数据线
- RES#:复位引脚,低电平有效
- D/C#:数据/命令控制引脚,决定发送的是命令还是显示数据
SSD1306支持三种接口方式:6800/8000系列并行接口、I2C接口或SPI接口。我们的0.96寸模块通常采用I2C接口,只需要两根信号线,极大地简化了连接。
芯片的初始化过程很有讲究。上电后,首先需要发送一系列配置命令:
- 关闭显示(AEH)
- 设置128x64显示模式
- 重置SEG和COM映射关系
- 清除串行接口中的移位寄存器数据
- 设置显示起始行
- 配置对比度和显示模式等
这些初始化步骤确保了屏幕能够正常工作和显示内容。我在实际项目中遇到过因为初始化顺序不对导致显示异常的情况,所以务必按照数据手册的推荐流程来操作。
2.2 GDDRAM结构与寻址机制
GDDRAM是SSD1306内部的显示数据RAM,它的大小为128x64位,正好对应屏幕的128列和64行像素。这片RAM被分成8个页面(Page),每个页面包含8行(对应COM0-COM7)和128列。
这种分页结构很巧妙:每个页面存储8行像素的数据,一个字节的数据对应8个垂直像素。当我们向GDDRAM写入一个字节时,字节的LSB(最低位)对应页面的最上方像素,MSB(最高位)对应最下方像素。
SSD1306支持三种寻址模式:
- 页寻址模式:最简单的方式,每次读写后列地址自动增加,适合字符显示
- 水平寻址模式:列地址到达终点后自动换行,适合连续数据写入
- 垂直寻址模式:行地址到达终点后自动换列,适合特定图形操作
在实际编程中,我最常用的是页寻址模式。比如要显示一个8x16的字符,需要先设置起始页地址和列地址,然后连续写入两个字节的数据(代表字符的上半部和下半部)。
// 设置光标位置(页地址和列地址)
void OLED_SetPos(uint8_t page, uint8_t col) {
OLED_WriteCmd(0xB0 + page); // 设置页地址
OLED_WriteCmd(col & 0x0F); // 设置列地址低4位
OLED_WriteCmd(0x10 + (col >> 4)); // 设置列地址高4位
}
理解GDDRAM的寻址机制非常重要,这是后续实现各种显示功能的基础。我曾经因为没搞清寻址模式,导致显示图像时出现错乱,花了半天时间才找到问题所在。
3. I2C通信协议:软件模拟实现详解
3.1 I2C基础与通信时序
I2C(Inter-Integrated Circuit)是一种串行通信总线,使用多主从架构。它只需要两根线:SDA(数据线)和SCL(时钟线)。所有设备都通过这两根线通信,每个设备有唯一的地址。
I2C的通信过程是这样的:
- 主机发送起始条件(START)
- 发送从机地址(7位地址+读写位)
- 从机应答(ACK)
- 传输数据字节
- 接收方应答
- 重复步骤4-5直到传输完成
- 主机发送停止条件(STOP)
起始条件和停止条件的定义很特别:
- 起始条件:SCL为高电平时,SDA从高变低
- 停止条件:SCL为高电平时,SDA从低变高
这种设计确保了总线状态的唯一性,不会与其他条件混淆。
为什么我们要用软件模拟I2C?因为STM32的硬件I2C有时会出现各种问题(特别是较早的型号),而软件模拟更加稳定和可移植。只要修改GPIO定义,就能在不同的MCU上使用相同的代码。
3.2 软件模拟I2C的实现
下面是我在实际项目中使用的软件I2C实现,稳定可靠:
// I2C起始信号
void OLED_I2C_Start(void) {
OLED_SCL_SET();
OLED_SDA_SET();
delay_us(5);
OLED_SDA_RESET();
delay_us(5);
OLED_SCL_RESET();
delay_us(5);
}
// I2C停止信号
void OLED_I2C_Stop(void) {
OLED_SCL_RESET();
OLED_SDA_RESET();
delay_us(5);
OLED_SCL_SET();
delay_us(5);
OLED_SDA_SET();
delay_us(5);
}
// 等待应答
uint8_t OLED_I2C_WaitAck(void) {
uint8_t ack = 0;
OLED_SCL_SET();
delay_us(5);
// 检查SDA电平
if (HAL_GPIO_ReadPin(OLED_I2C_SDA_GPIO, OLED_I2C_SDA_PIN) == GPIO_PIN_RESET) {
ack = 1; // 收到应答
}
OLED_SCL_RESET();
delay_us(5);
return ack;
}
// 写入一个字节
void OLED_I2C


1734

被折叠的 条评论
为什么被折叠?



