一、概述:
STC8H内置了一组硬IIC控制器,相对于传统的c51单片机来说这是一个非常实用的功能,很多外设的通信接口都是IIC标准,相对于软IIC来说硬IIC使用简单,无需考虑其具体实现,只需调用内部控制指令即可实现IIC控制,硬IIC最重要的优点还有传输速率高,传输可靠稳定。IIC的操作与SPI类似,同样有主从之分,通过不同的地址来区分不同的设备,一条IIC总线上可以挂载多个设备,同时还支持中断与唤醒。
SSD1306的OLED屏的操作码与SPI操作相同,在使用IIC通信时只要理清IIC写数据的格式即可简单的实现通信功能,由于OLED与MCU之间的通信只需考虑其单向通信就行,所以数据格式也不是很复杂。
二、知识链接:
1、I2C配置寄存器(I2CCFG):
用于初始化IIC,ENI2C=1使能I2C功能,MSSL=1设置为主机模式,MSSPEED=20设置I2C的传输速度。
2、I2C控制寄存器(I2CMSCR):
其中MSCMD为控制命令
3、I2C主机状态寄存器(I2CMSST):
其中MSIF为传输标志位,传输完成硬件置1,需要软件方式清0。
4、SSD1306 OLED屏时序:
传输数据格式:
起始信号+地址码+ACK+控制码+数据+停止码
地址码:0x78
控制码:读0x40,写0x00
三、实验平台搭建:
1、MCU:STC-打狗棒系列核心实验板 V2.3
2、实验板平台:德飞莱LY-51s
3、显示:SSD1306 0.96寸白色OLED屏(4脚IIC接口)
4、硬件连接表:
0.96寸4脚OLED接线图硬I2C接线表
SDL---------->P24
CLK---------->P25
VCC---------->+5V
GND---------->GND
四、测试源代码:
#include <STC8H.h>
#include "intrins.h"
#include "oledfont.h"
#define uint8_t unsigned char
#define u8 unsigned char
#define u16 unsigned int
#define uint32_t unsigned long
#define SIZE 16
#define XLevelL 0x00
#define XLevelH 0x10
#define Max_Column 128
#define Max_Row 64
#define Brightness 0xFF
#define X_WIDTH 128
#define Y_WIDTH 64
#define OLED_CMD 0 //写命令
#define OLED_DATA 1 //写数据
/*********************************
0.96寸4脚OLED接线图硬IIC接线表
SDA---------->P24
CLK---------->P25
VCC---------->+5V
GND---------->GND
***********************************/
void init_IO();//初始化IO
//硬件IIC
void init_IIC();//初始化硬IIC
void IIC_wait();//执行等待
void IIC_SendData(unsigned char dat);//发送数据
void IIC_RevAck();//接收ACK信号
void IIC_STOP();//停止信号
//OLED控制用函数
void OLED_Init(void);
void OLED_WR_Byte(unsigned char dat,unsigned char cmd);
void OLED_ShowChar(uint8_t x,uint8_t y,uint8_t chr);
void OLED_ShowString(uint8_t x,uint8_t y, uint8_t *p);
void OLED_Set_Pos(unsigned char x, unsigned char y);
void OLED_Clear(void);
void main()
{
P_SW2 |= 0x80; //扩展寄存器XFR访问使能
init_IO(); //初始化IO
init_IIC();//初始化硬IIC
OLED_Init();//初始化OLED
OLED_ShowString(0,0," MG LeiYang!");
OLED_ShowString(0,4," by lizhipeng!");
OLED_ShowString(0,2," OLED I2C Test!");
OLED_ShowString(0,6," 2023/08/09");
while(1);
}
void init_IO()
{
RSTCFG=0x50; //开启RST键进入ISP模式
P0M1 = 0x00; P0M0 = 0x00; //设置P0口为准双向口
P2M1 = 0x00; P2M0 = 0x00; //设置P1口为准双向口
P3M1 = 0x00; P3M0 = 0x00; //设置P3口为准双向口
}
void init_IIC()//初始化硬IIC
{
P_SW2|=0x10;//IIC切换至P25、P24
I2CCFG=0xe0;//ENI2C=1、MSSL=1、speed=20
}
void IIC_wait()//等待传输完成
{
while(!(I2CMSST&0x40));//等待IIC传输完成
I2CMSST&=~0x40;//清空标志位
}
void IIC_START()//起始信号
{
I2CMSCR=0x01;
IIC_wait();
}
void IIC_SendData(unsigned char dat)//发送数据
{
I2CTXD=dat;//加载数据
I2CMSCR=0x02;
IIC_wait();
}
void IIC_RevAck()//接收ACK信号
{
I2CMSCR=0x03;
IIC_wait();
}
void IIC_STOP()//停止信号
{
I2CMSCR=0x06;
IIC_wait();
}
//cmd:数据/命令标志 0,表示命令;1,表示数据;
void OLED_WR_Byte(unsigned char dat,unsigned char cmd)
{
IIC_START();//开始信号
IIC_SendData(0x78);//OLED屏地址码0x78
IIC_RevAck();
if(cmd==0)
{
IIC_SendData(0x00);//控制码-写入命令
IIC_RevAck();
}
else
{
IIC_SendData(0x40);//控制码-写入数据
IIC_RevAck();
}
IIC_SendData(dat);//写入数据
IIC_RevAck();
IIC_STOP();//停止信号
}
/***********************以下部分借鉴于网络*****************************/
//设置显示坐标位置
void OLED_Set_Pos(unsigned char x, unsigned char y)
{
OLED_WR_Byte(0xb0+y,OLED_CMD);
OLED_WR_Byte(((x&0xf0)>>4)|0x10,OLED_CMD);
OLED_WR_Byte((x&0x0f)|0x01,OLED_CMD);
}
//清屏函数,清完屏,整个屏幕是黑色的!和没点亮一样!!!
void OLED_Clear(void)
{
u8 i,n;
for(i=0;i<8;i++)
{
OLED_WR_Byte (0xb0+i,OLED_CMD); //设置页地址(0~7)
OLED_WR_Byte (0x00,OLED_CMD); //设置显示位置—列低地址
OLED_WR_Byte (0x10,OLED_CMD); //设置显示位置—列高地址
for(n=0;n<128;n++)OLED_WR_Byte(0,OLED_DATA);
} //更新显示
}
//在指定位置显示一个字符,包括部分字符
//x:0~127
//y:0~63
//mode:0,反白显示;1,正常显示
//size:选择字体 16/12
void OLED_ShowChar(u8 x,u8 y,u8 chr)
{
unsigned char c=0,i=0;
c=chr-' ';//得到偏移后的值
if(x>Max_Column-1){x=0;y=y+2;}
if(SIZE ==16)
{
OLED_Set_Pos(x,y);
for(i=0;i<8;i++)
OLED_WR_Byte(F8X16[c*16+i],OLED_DATA);
OLED_Set_Pos(x,y+1);
for(i=0;i<8;i++)
OLED_WR_Byte(F8X16[c*16+i+8],OLED_DATA);
}
else {
OLED_Set_Pos(x,y+1);
for(i=0;i<6;i++)
OLED_WR_Byte(F6x8[c][i],OLED_DATA);
}
}
//显示一个字符号串
void OLED_ShowString(u8 x,u8 y,u8 *chr)
{
unsigned char j=0;
while (chr[j]!='\0')
{ OLED_ShowChar(x,y,chr[j]);
x+=8;
if(x>120){x=0;y+=2;}
j++;
}
}
//初始化SSD1306
void OLED_Init(void)
{
OLED_WR_Byte(0xAE,OLED_CMD);//--turn off oled panel
OLED_WR_Byte(0x00,OLED_CMD);//---set low column address
OLED_WR_Byte(0x10,OLED_CMD);//---set high column address
OLED_WR_Byte(0x40,OLED_CMD);//--set start line address Set Mapping RAM Display Start Line (0x00~0x3F)
OLED_WR_Byte(0x81,OLED_CMD);//--set contrast control register
OLED_WR_Byte(0xCF,OLED_CMD); // Set SEG Output Current Brightness
OLED_WR_Byte(0xA1,OLED_CMD);//--Set SEG/Column Mapping 0xa0左右反置 0xa1正常
OLED_WR_Byte(0xC8,OLED_CMD);//Set COM/Row Scan Direction 0xc0上下反置 0xc8正常
OLED_WR_Byte(0xA6,OLED_CMD);//--set normal display
OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64)
OLED_WR_Byte(0x3f,OLED_CMD);//--1/64 duty
OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset Shift Mapping RAM Counter (0x00~0x3F)
OLED_WR_Byte(0x00,OLED_CMD);//-not offset
OLED_WR_Byte(0xd5,OLED_CMD);//--set display clock divide ratio/oscillator frequency
OLED_WR_Byte(0x80,OLED_CMD);//--set divide ratio, Set Clock as 100 Frames/Sec
OLED_WR_Byte(0xD9,OLED_CMD);//--set pre-charge period
OLED_WR_Byte(0xF1,OLED_CMD);//Set Pre-Charge as 15 Clocks & Discharge as 1 Clock
OLED_WR_Byte(0xDA,OLED_CMD);//--set com pins hardware configuration
OLED_WR_Byte(0x12,OLED_CMD);
OLED_WR_Byte(0xDB,OLED_CMD);//--set vcomh
OLED_WR_Byte(0x40,OLED_CMD);//Set VCOM Deselect Level
OLED_WR_Byte(0x20,OLED_CMD);//-Set Page Addressing Mode (0x00/0x01/0x02)
OLED_WR_Byte(0x02,OLED_CMD);//
OLED_WR_Byte(0x8D,OLED_CMD);//--set Charge Pump enable/disable
OLED_WR_Byte(0x14,OLED_CMD);//--set(0x10) disable
OLED_WR_Byte(0xA4,OLED_CMD);// Disable Entire Display On (0xa4/0xa5)
OLED_WR_Byte(0xA6,OLED_CMD);// Disable Inverse Display On (0xa6/a7)
OLED_WR_Byte(0xAF,OLED_CMD);//--turn on oled panel
OLED_WR_Byte(0xAF,OLED_CMD); /*display ON*/
OLED_Clear();
OLED_Set_Pos(0,0);
}