声明:本专题主要针对训练营的学生,因为我们知道你仍然不会使用一些常用的模块,所以我们开设了个专栏来解释如何快速启动一些模块在项目中,请注明转载的来源。 作者:渣渣鑫
目录
- 专题:OLED的使用说明
-
- OLED介绍
- 硬件配置
- 软件配置(IIC协议层)
- oled配置代码(这个可以直接移植)
- 简单说一下字模软件的使用(没有的找我要)
- 总结
- 点赞 再看是对我最大的鼓励!
专题:OLED的使用说明
OLED介绍
OLED,即有机发光二极管(Organic Light-Emitting Diode),也叫有机电激光显示(Organic Electroluminesence Display, OELD)。OLED由于具有自发光、无背光源、对比度高、厚度薄、视角广、反应速度快、柔性面板、使用温度范围广、结构简单、工艺简单等优良特点,被认为是下一代平面显示器的新兴应用技术。
OLED显示技术具有自发光的特点,采用非常薄的有机材料涂层和玻璃基板,当电流通过时,这些有机材料会发光,OLED从2003年开始,这种显示设备视角大,可以节约电能。MP应用于3播放器。
LCD都需要背光,而且OLED不需要,因为它是自发光的。这样同样的显示,OLED效果更好。以目前的技术,OLED尺寸难以大规模化,但分辨率确实很高。
硬件配置
主要使用的IIC总线上的SDA数据线和SCL时钟线,这里接上拉电阻的目的是当iic当总线处于空闲状态时,可以处于高电平状态。从图中可以看出iic总线支持多设备。
软件配置(IIC协议层)
在配置IIC在总线之前,我们需要先对应GPIO打开端口控制线
/*********************************** 重写IIC时序 *****************************************************/ //初始化GPIO /*SDA输出模式 PA3*/ void OLED_SDA_OUT(void) {
GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_OLED_SDA, ENABLE); /* 打开GPIO时钟 */ GPIO_InitStructure.GPIO_Pin = OLED_SDA_PIN; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD; /* 开漏输出 */ GPIO_Init(OLED_SDA_PORT, &GPIO_InitStructure); } /*SDA输入模式 PA3*/ void OLED_SDA_IN(void) {
GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_OLED_SDA, ENABLE); /* 打开GPIO时钟 */ GPIO_InitStructure.GPIO_Pin = OLED_SDA_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; /* // 浮空输入 */ GPIO_Init(OLED_SDA_PORT, &GPIO_InitStructure); }
先简单了解一下SDA与SCL的作用:当SCL高电平时,SDA上述数据必须保持稳定。SCL允许低电平时使用SDA线数据发生变化。
时序图必须要能看懂!
1、启动信号配置:SCL高电平期间,SDA电平由高到低跳转,我们的代码是模拟这个过程。
/*i2c 起始信号*/
void OLED_Start(void)
{
OLED_SDA_OUT(); //sda线输出
OLED_SCL_HIGH;
OLED_SDA_HIGH;
i2c_delay();
/*当CLK为高时,SDA由高变为低*/
OLED_SDA_L0W;
i2c_delay();
/*将SCL电位拉低,钳住SCL线,准备发送地址数据*/
OLED_SCL_L0W;
}
2、停止信号:当SCL为高电平期间,SDA由低电平变为高电平
/*i2c 停止信号*/
void OLED_Stop(void)
{
OLED_SDA_OUT(); //sda线输出
/*scl为高电平时,SDA由低电平变为高电平*/
OLED_SDA_L0W;
OLED_SCL_L0W;
i2c_delay();
OLED_SCL_HIGH;
OLED_SDA_HIGH;
i2c_delay();
}
3、空闲状态:当IIC总线的数据线SDA和时钟线SCL两条信号线同时处于高电平时,规定为总线的空闲状态。此时各个器件的输出级场效应管均处在截止状态,即释放总线,由两条信号线各自的上拉电阻把电平拉高。
4、应答信号:发送器每发送一个字节(8个bit),就在时钟脉冲9期间释放数据线,由接收器反馈一个应答信号。 应答信号为低电平时,规定为位(,简称应答位),表示接收器已经成功地接收了该字节; 应答信号为高电平时,规定为(),一般表示接收器接收该字节没有成功。 检测应答:
-
SCL为高电平的时候可以读取SDA的状态,因此可以将SDA模式切换为输入模式,读取SDA引脚状态,0位应答,1位非应答
-
SCL为低电平允许数据发生变化
-
SDA:为高电平的时候可以占用总线,此时将SDA拉低,开始通信。当为低电平的时候,SDA已经被占用。 SCL: SCL为高电平的时候要求数据稳定 SCL为低电平的时候允许数据改变
等待应答的作用就是来判断SDA传给的是低电平还是高电平。
/*等待应答信号到来 (有效应答:从机第9个 SCL=0 时 SDA 被从机拉低, 并且 SCL = 1时 SDA依然为低) 返回值: 1,接收应答失败 0,接收应答成功*/
uint8_t OLED_WaitAck(void)
{
uint8_t ucErrTime=0;
OLED_SDA_IN(); //SDA设置为输入模式 (从机给一个低电平做为应答)
OLED_SDA_HIGH;i2c_delay();
OLED_SCL_HIGH;i2c_delay();
//读取SDA的电平状态
while(OLED_SDA_Read)
{
ucErrTime++;
if(ucErrTime>250)
{
OLED_Stop();
return 1;//接收应答失败
}
}
OLED_SCL_L0W; //时钟输出0
return 0; //接收应答成功
}
有效应答和无效应答:
void OLED_Ack(void)
{
OLED_SCL_L0W;
OLED_SDA_OUT();
OLED_SDA_L0W;
i2c_delay();
OLED_SCL_HIGH;
i2c_delay();
OLED_SCL_L0W;
}
void OLED_NAck(void)
{
OLED_SCL_L0W;
OLED_SDA_OUT();
OLED_SDA_HIGH;
i2c_delay();
OLED_SCL_HIGH;
i2c_delay();
OLED_SCL_L0W;
}
5、字节格式: SDA数据线上的每个字节必须是8位,每次传输的字节数量没有限制。。首先传输的数据是最高位(MSB),SDA上的数据必须在SCL高电平周期时保持稳定,数据的高低电平翻转变化发生在SCL低电平时期。
-
每一个字节后面跟着一个ACK,有ACK就可以继续写或读。NACK,就停止
-
ACK:主机释放总线,传输完字节最后1位后的SCL的高电处,从机拉低电平。
-
NACK:主机释放总线,传输完字节最后1位后的SCL的高电处,从机无响应,总线为高电平。
发送数据:
/****************************************************************************** *函 数:void IIC_SendByte(uint8_t txd) *功 能:IIC发送一个字节 *参 数:data 要写的数据 *返回值:无 *备 注:主机往从机发 *******************************************************************************/
void OLED_SendByte(uint8_t data)
{
uint8_t t;
OLED_SDA_OUT();
OLED_SCL_L0W; //拉低时钟开始数据传输
for(t=0;t<8;t++)
{
if((data&0x80)>>7)//0x80=1000 0000
OLED_SDA_HIGH;
else
OLED_SDA_L0W;
data<<=1;
i2c_delay();
OLED_SCL_HIGH;
i2c_delay();
OLED_SCL_L0W;
i2c_delay();
}
}
读取数据:
uint8_t OLED_ReadByte(uint8_t ack)
{
uint8_t i,receive=0;
OLED_SDA_IN(); //SDA设置为输入模式 等待接收从机返回数据
for(i=0;i<8;i++ )
{
OLED_SCL_L;
i2c_delay();
OLED_SCL_H;
receive<<=1;
if(OLED_SDA_Read)receive++; //从机发送的电平
i2c_delay();
}
if(ack)
OLED_Ack(); //发送ACK
else
OLED_NAck(); //发送nACK
return receive;
}
总结:最难的时序部分代码完成后,就可以使用模拟的iic代码来完成oled屏幕代码的初始化了
oled配置代码(这个大家直接移植就行)
//初始化SSD1306 // 端口用PA5,PA6 void OLED_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //使能A端口时钟 GPIO_InitStructure.GPIO_Pin = OLED_SCL_PIN|OLED_SDA_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//速度50MHz GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化 GPIO_SetBits(GPIOA,OLED_SCL_PIN|OLED_SDA_PIN); delay_ms(800); OLED_WR_Byte(0xAE,OLED_CMD);//--display off 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 OLED_WR_Byte(0xB0,OLED_CMD);//--set page address OLED_WR_Byte(0x81,OLED_CMD); // contract control OLED_WR_Byte(0xFF,OLED_CMD);//--128 OLED_WR_Byte(0xA1,OLED_CMD);//set segment remap OLED_WR_Byte(0xA6,OLED_CMD);//--normal / reverse OLED_WR_Byte(0xA8,OLED_CMD);//--set multiplex ratio(1 to 64) OLED_WR_Byte(0x3F,OLED_CMD);//--1/32 duty OLED_WR_Byte(0xC8,OLED_CMD);//Com scan direction OLED_WR_Byte(0xD3,OLED_CMD);//-set display offset OLED_WR_Byte(0x00,OLED_CMD);// OLED_WR_Byte(0xD5,OLED_CMD);//set osc division OLED_WR_Byte(0x80,OLED_CMD);// OLED_WR_Byte(0xD8,OLED_CMD);//set area color mode off OLED_WR_Byte(0x05,OLED_CMD);// OLED_WR_Byte(0xD9,OLED_CMD);//Set Pre-Charge Period OLED_WR_Byte(0xF1,OLED_CMD);// OLED_WR_Byte(0xDA,OLED_CMD);//set com pin configuartion OLED_WR_Byte(0x12,OLED_CMD);// OLED_WR_Byte(0xDB,OLED_CMD);//set Vcomh OLED_WR_Byte(0x30,OLED_CMD);// OLED_WR_Byte(0x8D,OLED_CMD);//set charge pump enable OLED_WR_Byte(0x14,OLED_CMD);// OLED_WR_Byte(0xAF,OLED_CMD);//--turn on oled panel } /******************************************** // fill_Picture ********************************************/ void fill_picture(unsigned char fill_Data) { unsigned char m,n; for(m=0;m<8;m++) { OLED_WR_Byte(0xb0+m,0); //page0-page1 OLED_WR_Byte(0x00,0); //low column start address OLED_WR_Byte(0x10,0); //high column start address for(n=0;n<128;n++) { OLED_WR_Byte(fill_Data,1); } } } /***********************Delay****************************************/ void Delay_50ms(unsigned int Del_50ms) { unsigned int m; for(;Del_50ms>0;Del_50ms--) for(m=6245;m>0;m--); } void Delay_1ms(unsigned int Del_1ms) { unsigned char j; while(Del_1ms--) { for(j=0;j<123;j++); } } //坐标设置 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),OLED_CMD); } //开启OLED显示 void OLED_Display_On(void) { OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC命令 OLED_WR_Byte(0X14,OLED_CMD); //DCDC ON OLED_WR_Byte(0XAF,OLED_CMD); //DISPLAY ON } //关闭OLED显示 void OLED_Display_Off(void) { OLED_WR_Byte(0X8D,OLED_CMD); //SET DCDC命令 OLED_WR_Byte(0X10,OLED_CMD); //DCDC OFF OLED_WR_Byte(0XAE,OLED_CMD); //DISPLAY OFF } //清屏函数,清完屏,整个屏幕是黑色的!和没点亮一样!!! 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); } //更新显示 } void OLED_On(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(1,OLED_DATA); } //更新显示 } //在指定位置显示一个字符,包括部分字符 //x:0~127 //y:0~63 //mode:0,反白显示;1,正常显示 //size:选择字体 16/12 void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 Char_Size) { unsigned char c=0,i=0; c=chr-' ';//得到偏移后的值 if(x>Max_Column-1){ x=0;y=y+2;} if(Char_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); for(i=0;i<6;i++) OLED_WR_Byte(F6x8[c][i],OLED_DATA); } } //m^n函数 u32 oled_pow(u8 m,u8 n) { u32 result=1; while(n--)result*=m; return result; } //显示2个数字 //x,y :起点坐标 //len :数字的位数 //size:字体大小 //mode:模式 0,填充模式;1,叠加模式 //num:数值(0~4294967295); void OLED_ShowNum(u8 x,u8 y,u32 num,u8 len,u8 size2) { u8 t,temp; u8 enshow=0; for(t=0;t<len;t++) { temp=(num/oled_pow(10,len-t-1))%10; if(enshow==0&&t<(len-1)) { if(temp==0) { OLED_ShowChar(x+(size2/2)*t,y,' ',size2); continue; }else enshow=1; } OLED_ShowChar(x+(size2/2)*t,y,temp+'0',size2); } } void OLED_Float(unsigned char Y,unsigned char X,double real,unsigned char N) { unsigned char i_Count=1; unsigned char n[12]={ 0}; long j=1; int real_int=0; double decimal=0; unsigned int real_decimal=0; if(real<0) { real_int=(int)(-real); } else { real_int=(int)real; } decimal=real-real_int; real_decimal=decimal*1e4; while(real_int/10/j!=0) { j=j*10;i_Count++; } n[0]=(real_int/10000)%10; n[1]=(real_int/1000)%10; n[2]=(real_int/100)%10; n[3]=(real_int/10)%10; n[4]=(real_int/1)%10; n[5]='.'; n[ 标签:
二极管场效应模块二极管w0508sa150电位器turn