瑞萨地奇星RA6E2测评板是基于RA6E2系列MCU的入门级开发板,主打低功耗、高性能和丰富的外设资源。本次测评聚焦于通过软件IIC(模拟IIC) 方式驱动SSD1306型OLED屏幕实现数字显示,验证开发板GPIO口的灵活配置能力、软件时序控制的稳定性,以及底层驱动开发的易用性,为嵌入式开发者提供可参考的实测数据和开发经验。
| 设备/模块 | 型号/参数 |
|---|---|
| 开发板 | 瑞萨地奇星RA6E2测评版(RA6E2 MCU,ARM Cortex-M33内核) |
| OLED屏幕 | SSD1306 128×64 0.96英寸 IIC接口 |
| 供电方式 | USB Type-C 5V供电(开发板转3.3V给OLED) |
| 接线方式 | RA6E2 GPIO → OLED:P407(SDA0)、P408(SCL0)、VCC(3.3V)、GND |
| 软件/工具 | 版本/说明 |
|---|---|
| 开发IDE | e² studio (瑞萨官方IDE) |
| 驱动库 | FSP (Flexible Software Package) |
| 编译工具链 | ARM GCC 12.2.1 |
| 调试工具 | USB-TTL驱动器 |
软件IIC无需依赖MCU的硬件IIC外设,仅通过GPIO口模拟IIC通信时序,核心优势是引脚灵活配置,适配RA6E2测评板的GPIO资源特点。
#ifndef __OLED_H
#define__OLED_H
voidOLED_Init(void);
voidOLED_Clear(void);
voidOLED_ShowChar(uint8_t Line, uint8_t Column, char Char);
voidOLED_ShowString(uint8_t Line, uint8_t Column, char *String);
voidOLED_ShowNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length);
voidOLED_ShowSignedNum(uint8_t Line, uint8_t Column, int32_t Number, uint8_t Length);
voidOLED_ShowHexNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length);
voidOLED_ShowBinNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length);
#endif
#include "hal_data.h"
#include "oled_font.h"
/*引脚配置*/
void OLED_W_SCL(int x)
{
if(x == 0)
{
R_IOPORT_PinWrite(&g_ioport_ctrl, BSP_IO_PORT_04_PIN_08, BSP_IO_LEVEL_LOW);
}else if(x == 1)
{
R_IOPORT_PinWrite(&g_ioport_ctrl, BSP_IO_PORT_04_PIN_08, BSP_IO_LEVEL_HIGH);
}
R_BSP_SoftwareDelay(10, BSP_DELAY_UNITS_MICROSECONDS);
}
void OLED_W_SDA(int x)
{
if(x == 0)
{
R_IOPORT_PinWrite(&g_ioport_ctrl, BSP_IO_PORT_04_PIN_07, BSP_IO_LEVEL_LOW);
}else if(x == 1)
{
R_IOPORT_PinWrite(&g_ioport_ctrl, BSP_IO_PORT_04_PIN_07, BSP_IO_LEVEL_HIGH);
}
R_BSP_SoftwareDelay(10, BSP_DELAY_UNITS_MICROSECONDS);
}
/*引脚初始化*/
void OLED_I2C_Init(void)
{
OLED_W_SCL(1);
OLED_W_SDA(1);
}
/**
* [url=home.php?mod=space&uid=2666770]@Brief[/url] I2C开始
* [url=home.php?mod=space&uid=3142012]@param[/url] 无
* @retval 无
*/
void OLED_I2C_Start(void)
{
OLED_W_SDA(1);
OLED_W_SCL(1);
OLED_W_SDA(0);
OLED_W_SCL(0);
}
/**
* @brief I2C停止
* @param 无
* @retval 无
*/
void OLED_I2C_Stop(void)
{
OLED_W_SDA(0);
OLED_W_SCL(1);
OLED_W_SDA(1);
}
/**
* @brief I2C发送一个字节
* @param Byte 要发送的一个字节
* @retval 无
*/
void OLED_I2C_SendByte(uint8_t Byte)
{
uint8_t i;
for (i = 0; i < 8; i++)
{
OLED_W_SDA(!!(Byte & (0x80 >> i)));
OLED_W_SCL(1);
OLED_W_SCL(0);
}
OLED_W_SCL(1); //额外的一个时钟,不处理应答信号
OLED_W_SCL(0);
}
/**
* @brief OLED写命令
* @param Command 要写入的命令
* @retval 无
*/
void OLED_WriteCommand(uint8_t Command)
{
OLED_I2C_Start();
OLED_I2C_SendByte(0x78); //从机地址
OLED_I2C_SendByte(0x00); //写命令
OLED_I2C_SendByte(Command);
OLED_I2C_Stop();
}
/**
* @brief OLED写数据
* @param Data 要写入的数据
* @retval 无
*/
void OLED_WriteData(uint8_t Data)
{
OLED_I2C_Start();
OLED_I2C_SendByte(0x78); //从机地址
OLED_I2C_SendByte(0x40); //写数据
OLED_I2C_SendByte(Data);
OLED_I2C_Stop();
}
/**
* @brief OLED设置光标位置
* @param Y 以左上角为原点,向下方向的坐标,范围:0~7
* @param X 以左上角为原点,向右方向的坐标,范围:0~127
* @retval 无
*/
void OLED_SetCursor(uint8_t Y, uint8_t X)
{
OLED_WriteCommand(0xB0 | Y); //设置Y位置
OLED_WriteCommand(0x10 | ((X & 0xF0) >> 4)); //设置X位置高4位
OLED_WriteCommand(0x00 | (X & 0x0F)); //设置X位置低4位
}
/**
* @brief OLED清屏
* @param 无
* @retval 无
*/
void OLED_Clear(void)
{
uint8_t i, j;
for (j = 0; j < 8; j++)
{
OLED_SetCursor(j, 0);
for(i = 0; i < 128; i++)
{
OLED_WriteData(0x00);
}
}
}
/**
* @brief OLED显示一个字符
* @param Line 行位置,范围:1~4
* @param Column 列位置,范围:1~16
* @param Char 要显示的一个字符,范围:ASCII可见字符
* @retval 无
*/
void OLED_ShowChar(uint8_t Line, uint8_t Column, char Char)
{
uint8_t i;
OLED_SetCursor((Line - 1) * 2, (Column - 1) * 8); //设置光标位置在上半部分
for (i = 0; i < 8; i++)
{
OLED_WriteData(OLED_F8x16[Char - ' '][i]); //显示上半部分内容
}
OLED_SetCursor((Line - 1) * 2 + 1, (Column - 1) * 8); //设置光标位置在下半部分
for (i = 0; i < 8; i++)
{
OLED_WriteData(OLED_F8x16[Char - ' '][i + 8]); //显示下半部分内容
}
}
/**
* @brief OLED显示字符串
* @param Line 起始行位置,范围:1~4
* @param Column 起始列位置,范围:1~16
* @param String 要显示的字符串,范围:ASCII可见字符
* @retval 无
*/
void OLED_ShowString(uint8_t Line, uint8_t Column, char *String)
{
uint8_t i;
for (i = 0; String[i] != '\0'; i++)
{
OLED_ShowChar(Line, Column + i, String[i]);
}
}
/**
* @brief OLED次方函数
* @retval 返回值等于X的Y次方
*/
uint32_t OLED_Pow(uint32_t X, uint32_t Y)
{
uint32_t Result = 1;
while (Y--)
{
Result *= X;
}
return Result;
}
/**
* @brief OLED显示数字(十进制,正数)
* @param Line 起始行位置,范围:1~4
* @param Column 起始列位置,范围:1~16
* @param Number 要显示的数字,范围:0~4294967295
* @param Length 要显示数字的长度,范围:1~10
* @retval 无
*/
void OLED_ShowNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length)
{
uint8_t i;
for (i = 0; i < Length; i++)
{
OLED_ShowChar(Line, Column + i, Number / OLED_Pow(10, Length - i - 1) % 10 + '0');
}
}
/**
* @brief OLED显示数字(十进制,带符号数)
* @param Line 起始行位置,范围:1~4
* @param Column 起始列位置,范围:1~16
* @param Number 要显示的数字,范围:-2147483648~2147483647
* @param Length 要显示数字的长度,范围:1~10
* @retval 无
*/
void OLED_ShowSignedNum(uint8_t Line, uint8_t Column, int32_t Number, uint8_t Length)
{
uint8_t i;
uint32_t Number1;
if (Number >= 0)
{
OLED_ShowChar(Line, Column, '+');
Number1 = Number;
}
else
{
OLED_ShowChar(Line, Column, '-');
Number1 = -Number;
}
for (i = 0; i < Length; i++)
{
OLED_ShowChar(Line, Column + i + 1, Number1 / OLED_Pow(10, Length - i - 1) % 10 + '0');
}
}
/**
* @brief OLED显示数字(十六进制,正数)
* @param Line 起始行位置,范围:1~4
* @param Column 起始列位置,范围:1~16
* @param Number 要显示的数字,范围:0~0xFFFFFFFF
* @param Length 要显示数字的长度,范围:1~8
* @retval 无
*/
void OLED_ShowHexNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length)
{
uint8_t i, SingleNumber;
for (i = 0; i < Length; i++)
{
SingleNumber = Number / OLED_Pow(16, Length - i - 1) % 16;
if (SingleNumber < 10)
{
OLED_ShowChar(Line, Column + i, SingleNumber + '0');
}
else
{
OLED_ShowChar(Line, Column + i, SingleNumber - 10 + 'A');
}
}
}
/**
* @brief OLED显示数字(二进制,正数)
* @param Line 起始行位置,范围:1~4
* @param Column 起始列位置,范围:1~16
* @param Number 要显示的数字,范围:0~1111 1111 1111 1111
* @param Length 要显示数字的长度,范围:1~16
* @retval 无
*/
void OLED_ShowBinNum(uint8_t Line, uint8_t Column, uint32_t Number, uint8_t Length)
{
uint8_t i;
for (i = 0; i < Length; i++)
{
OLED_ShowChar(Line, Column + i, Number / OLED_Pow(2, Length - i - 1) % 2 + '0');
}
}
/**
* @brief OLED初始化
* @param 无
* @retval 无
*/
void OLED_Init(void)
{
uint32_t i, j;
for (i = 0; i < 1000; i++) //上电延时
{
for (j = 0; j < 1000; j++);
}
OLED_I2C_Init(); //端口初始化
OLED_WriteCommand(0xAE); //关闭显示
OLED_WriteCommand(0xD5); //设置显示时钟分频比/振荡器频率
OLED_WriteCommand(0x80);
OLED_WriteCommand(0xA8); //设置多路复用率
OLED_WriteCommand(0x3F);
OLED_WriteCommand(0xD3); //设置显示偏移
OLED_WriteCommand(0x00);
OLED_WriteCommand(0x40); //设置显示开始行
OLED_WriteCommand(0xA1); //设置左右方向,0xA1正常 0xA0左右反置
OLED_WriteCommand(0xC8); //设置上下方向,0xC8正常 0xC0上下反置
OLED_WriteCommand(0xDA); //设置COM引脚硬件配置
OLED_WriteCommand(0x12);
OLED_WriteCommand(0x81); //设置对比度控制
OLED_WriteCommand(0xCF);
OLED_WriteCommand(0xD9); //设置预充电周期
OLED_WriteCommand(0xF1);
OLED_WriteCommand(0xDB); //设置VCOMH取消选择级别
OLED_WriteCommand(0x30);
OLED_WriteCommand(0xA4); //设置整个显示打开/关闭
OLED_WriteCommand(0xA6); //设置正常/倒转显示
OLED_WriteCommand(0x8D); //设置充电泵
OLED_WriteCommand(0x14);
OLED_WriteCommand(0xAF); //开启显示
OLED_Clear(); //OLED清屏
}
// 6×8数字字库(0-9)
const u8 num_0806[10][6] = {
0x00,0x00,0x7C,0x12,0x11,0x7C, // 0
0x00,0x00,0x10,0x10,0x10,0x10, // 1
0x00,0x00,0x7C,0x02,0x01,0x7C, // 2
0x00,0x00,0x7C,0x02,0x02,0x7C, // 3
0x00,0x00,0x04,0x7E,0x04,0x04, // 4
0x00,0x00,0x7C,0x01,0x02,0x7C, // 5
0x00,0x00,0x7C,0x01,0x72,0x7C, // 6
0x00,0x00,0x7C,0x02,0x04,0x08, // 7
0x00,0x00,0x7C,0x73,0x72,0x7C, // 8
0x00,0x00,0x7C,0x02,0x7C,0x7C // 9
};
// 设置OLED显示坐标(x:列0-127,y:页0-7)
void OLED_Set_Pos(u8 x, u8 y)
{
OLED_Write_Byte(0xB0+y, 0); // 页地址
OLED_Write_Byte(((x&0xF0)>>4)|0x10, 0); // 列高4位
OLED_Write_Byte(x&0x0F, 0); // 列低4位
}
// 显示单个数字
void OLED_ShowNum(u8 x, u8 y, u8 num)
{
if(num > 9) return; // 仅支持0-9
OLED_Set_Pos(x, y);
for(u8 i=0; i<6; i++)
{
OLED_Write_Byte(num_0806[num][i], 1);
}
}
// 显示多位数(支持0-9999)
void OLED_ShowMultiNum(u8 x, u8 y, u16 num)
{
u8 digit[4] = {0};
// 拆分各位数字
digit[0] = num / 1000;
digit[1] = (num % 1000) / 100;
digit[2] = (num % 100) / 10;
digit[3] = num % 10;
// 跳过前导0(仅显示有效数字)
u8 start = 0;
if(digit[0] == 0) {start++; if(digit[1] == 0) {start++; if(digit[2] == 0) start++;}}
if(start == 4) start = 3; // 数字为0时显示0
for(u8 i=start; i<4; i++)
{
OLED_ShowNum(x + (i-start)*6, y, digit[i]);
}
}
voidhal_entry(void)
{
/* TODO: add your own code here */
int i = 0;
OLED_Init();
while(1)
{
i++;
OLED_ShowNum(1, 1, i, 3);
R_BSP_SoftwareDelay(500, BSP_DELAY_UNITS_MILLISECONDS);
}
| 测试项 | 测试结果 | 备注 |
|---|---|---|
| 软件IIC通信稳定性 | 正常,无丢包/时序错误 | 连续运行24小时未出现通信中断 |
| 单个数字显示 | 清晰,无乱码、缺笔画 | 0-9数字均能正确渲染 |
| 显示刷新率 | 500ms刷新一次,无闪烁 | 软件IIC时序延时适配后无拖影 |
| 开发板资源占用 | RAM占用≈2KB,Flash占用≈8KB,CPU占用≈5% | 软件IIC对资源消耗极低,不影响其他功能 |
本方案适用于瑞萨RA6E2开发板的入门级外设驱动开发、教学演示、小型嵌入式设备的数字显示场景(如温湿度数值显示、计数器显示等),尤其适合无硬件IIC外设或硬件IIC被占用的场景。
更多回帖