天天看点

STM32f429使用0.96 oled

我的32单片机因为型号问题,和网络上的很多型号都有些不同,因此需要自己编写iic的读写数据、读写命令的代码来操作oled,因此下面的代码可以分为两部分,一部分是iic的读写,读写我是根据老师给的代码进行修改的,而oled的显示是根据正点原子提供的相关代码,至于将二者合二为一则是参考了网络上的一篇博客,网址也忘了,可能是自己把它弄成功之后太高兴的缘故。

这是我成功之后的图片,当然并不是下面的代码直接运行出来的结果!下面代码仅供参考。

STM32f429使用0.96 oled
#ifndef _LSOLED_H_
#define _LSOLED_H_

#include "msys.h"                                 

#define OLED_SCLK_Clr() GPIO_ResetBits(GPIOA,GPIO_Pin_6)//SCL
#define OLED_SCLK_Set() GPIO_SetBits(GPIOA,GPIO_Pin_6)

#define OLED_SDIN_Clr() GPIO_ResetBits(GPIOA,GPIO_Pin_5)//SDA
#define OLED_SDIN_Set() GPIO_SetBits(GPIOA,GPIO_Pin_5)

#define OLED_CMD 0	//写命令
#define OLED_DATA 1	//写数据


#define GPIO_PORT_I2C	GPIOB			/* GPIO端口 */
#define RCC_I2C_PORT 	RCC_AHB1Periph_GPIOB		/* GPIO端口时钟 */
#define I2C_SCL_PIN		GPIO_Pin_6			/* 连接到SCL时钟线的GPIO */
#define I2C_SDA_PIN		GPIO_Pin_7			/* 连接到SDA数据线的GPIO */

#define I2C_SCL_1()  GPIO_PORT_I2C->BSRRL = I2C_SCL_PIN				/* SCL = 1 */
#define I2C_SCL_0()  GPIO_PORT_I2C->BSRRH = I2C_SCL_PIN				/* SCL = 0 */

#define I2C_SDA_1()  GPIO_PORT_I2C->BSRRL = I2C_SDA_PIN				/* SDA = 1 */
#define I2C_SDA_0()  GPIO_PORT_I2C->BSRRH = I2C_SDA_PIN				/* SDA = 0 */

#define I2C_SDA_READ() ((GPIO_PORT_I2C->IDR & I2C_SDA_PIN) != 0)	/* 读SDA口线状态 */
#define I2C_SCL_READ()  ((GPIO_PORT_I2C->IDR & I2C_SCL_PIN) != 0)	/* 读SCL口线状态 */


void bsp_InitI2C(void);
void i2c_Stop(void);
void i2c_Start(void);
static void i2c_Delay(void);
void i2c_SendByte(uint8_t _ucByte);
uint8_t i2c_ReadByte(void);
uint8_t i2c_WaitAck(void);


void Write_IIC_Command(unsigned char IIC_Command);         /*写命令*/
void Write_IIC_Data(unsigned char IIC_Data);               /*写数据*/
void fill_picture(unsigned char fill_Data);
void OLED_WR_Byte(unsigned dat,unsigned cmd);
void OLED_Display_On(void);
void OLED_Display_Off(void);

void OLED_Init(void);
void OLED_Display_Off(void);
void OLED_Display_On(void);
void OLED_Clear(void);
void OLED_Refresh_Gram(void);
void OLED_ShowString(u8 x,u8 y,const u8 *p,u8 size);
void OLED_ShowNum(u8 x,u8 y,u32 num,u8 len,u8 size);
void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 size,u8 mode);
void OLED_ShowSideRectangle(void);
void OLED_ShowSnake(uint16_t footx,uint16_t footy,uint8_t *flag);

#endif
           
#include "lsoled.h"                               //头文件                                           
#include "delay.h"                                   //延时
#include "lsoledfront.h"                          //字体,我用的是正电原子提供的

u8 OLED_GRAM[128][8];

void bsp_InitI2C(void)                            //iic初始化
{
	GPIO_InitTypeDef GPIO_InitStructure;

	RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);	/* 打开GPIO时钟 */

	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;		/* 设为输出口 */
	GPIO_InitStructure.GPIO_OType = GPIO_OType_OD;		/* 设为开漏模式 */
	GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;	/* 上下拉电阻不使能 */
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_25MHz;	/* IO口最大速度 */

	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;
	GPIO_Init(GPIOB, &GPIO_InitStructure);

	/* 给一个停止信号, 复位I2C总线上的所有设备到待机模式 */
	i2c_Stop();
}

static void i2c_Delay(void)
{
	uint8_t i;

	/* 
		CPU主频168MHz时,在内部Flash运行, MDK工程不优化。用台式示波器观测波形。
		循环次数为5时,SCL频率 = 1.78MHz (读耗时: 92ms, 读写正常,但是用示波器探头碰上就读写失败。时序接近临界)
		循环次数为10时,SCL频率 = 1.1MHz (读耗时: 138ms, 读速度: 118724B/s)
		循环次数为30时,SCL频率 = 440KHz, SCL高电平时间1.0us,SCL低电平时间1.2us

		上拉电阻选择2.2K欧时,SCL上升沿时间约0.5us,如果选4.7K欧,则上升沿约1us

		实际应用选择400KHz左右的速率即可
	*/
	for (i = 0; i < 30; i++);
}

void i2c_Stop(void)
{
	/* 当SCL高电平时,SDA出现一个上跳沿表示I2C总线停止信号 */
	I2C_SDA_0(); //SDA = 0;
	I2C_SCL_1(); //SCL = 0;
	i2c_Delay(); //延时
	I2C_SDA_1(); //SDA = 1;
	i2c_Delay();
}

void i2c_Start(void)
{
	/* 当SCL高电平时,SDA出现一个下跳沿表示I2C总线启动信号 */
	I2C_SDA_1(); //SDA = 1;
	I2C_SCL_1(); //SCL = 1;
	i2c_Delay(); //延时
	I2C_SDA_0(); //SDA = 0;
	i2c_Delay(); //延时
	
	I2C_SCL_0(); //SCL = 0;
	i2c_Delay();
}


void Write_IIC_Data(unsigned char IIC_Data)
{
	i2c_Start();
	i2c_SendByte(0x78);	//D/C#=0; R/W#=0
	i2c_WaitAck();	
	i2c_SendByte(0x40);	//write data
	i2c_WaitAck();	
	i2c_SendByte(IIC_Data);
	i2c_WaitAck();	
	i2c_Stop();
}

void OLED_WR_Byte(unsigned dat,unsigned cmd)
{
	if(cmd)
	{
		Write_IIC_Data(dat);
	}
	else 
	{
		Write_IIC_Command(dat);	
	}
}

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);
		}
	} //更新显示
}

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);
		}
	}
}

void Write_IIC_Command(unsigned char IIC_Command)
{
	i2c_Start();
	i2c_SendByte(0x78); //Slave address,SA0=0
	i2c_WaitAck();	
	i2c_SendByte(0x00);	//write command
	i2c_WaitAck();	
	i2c_SendByte(IIC_Command);
	i2c_WaitAck();	
	i2c_Stop();
}

void i2c_SendByte(uint8_t _ucByte)  //主机向从机(EEPROM)发送地址或数据函数
{
	uint8_t i;

	/* 先发送字节的高位bit7 */
	for (i = 0; i < 8; i++) //一个时钟周期发送一个位,所以需要8个时钟周期
	{
		if (_ucByte & 0x80)  //获取最高位的值
		{
			I2C_SDA_1();
		}
		else
		{
			I2C_SDA_0();
		}
		i2c_Delay();
		I2C_SCL_1();
		i2c_Delay();
		I2C_SCL_0();
		if (i == 7)
		{
			 I2C_SDA_1(); // 释放总线
		}
		_ucByte <<= 1;	/* 左移一个bit */
		i2c_Delay();
	}
}

uint8_t i2c_ReadByte(void)
{
	uint8_t i;
	uint8_t value;

	/* 读到第1个bit为数据的bit7 */
	value = 0;
	for (i = 0; i < 8; i++)
	{
		value <<= 1;
		I2C_SCL_1();
		i2c_Delay();
		if (I2C_SDA_READ())  //如果读到高电平,加1
		{
			value++; //SDA为1,高电平 
		}
		I2C_SCL_0();
		i2c_Delay();
	}
	return value;  //返回读到的一个字节数据
}

uint8_t i2c_WaitAck(void)
{
	uint8_t re;

	I2C_SDA_1();	/* CPU释放SDA总线 */
	i2c_Delay();
	I2C_SCL_1();	/* CPU驱动SCL = 1, 此时器件会返回ACK应答 */
	i2c_Delay();
	if (I2C_SDA_READ())	/* CPU读取SDA口线状态 */
	{
		re = 1; //SDA ==1 ,未收到应答信号
	}
	else
	{
		re = 0; //SDA == 0 收到应答信号
	}
	I2C_SCL_0();
	i2c_Delay();
	return re;
}

void OLED_Init(void)
{
		GPIO_InitTypeDef GPIO_InitStruct;
	
		RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);

		GPIO_InitStruct.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_6;  /*初始化PA5、PA6*/
		GPIO_InitStruct.GPIO_Mode =  GPIO_Mode_OUT;
		GPIO_InitStruct.GPIO_OType = GPIO_OType_PP;
		GPIO_InitStruct.GPIO_PuPd = GPIO_PuPd_UP;
		GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
	
		GPIO_Init(GPIOA,&GPIO_InitStruct);
		GPIO_SetBits(GPIOA,GPIO_Pin_5 | GPIO_Pin_6);
	
		delay_ms(800);
		
		OLED_WR_Byte(0xAE,OLED_CMD); //关闭显示
		OLED_WR_Byte(0xD5,OLED_CMD); //设置时钟分频因子,震荡频率
		OLED_WR_Byte(80,OLED_CMD);   //[3:0],分频因子;[7:4],震荡频率
		OLED_WR_Byte(0xA8,OLED_CMD); //设置驱动路数
		OLED_WR_Byte(0X3F,OLED_CMD); //默认0X3F(1/64) 
		OLED_WR_Byte(0xD3,OLED_CMD); //设置显示偏移
		OLED_WR_Byte(0X00,OLED_CMD); //默认为0
		OLED_WR_Byte(0x40,OLED_CMD); //设置显示开始行 [5:0],行数.										    
		OLED_WR_Byte(0x8D,OLED_CMD); //电荷泵设置
		OLED_WR_Byte(0x14,OLED_CMD); //bit2,开启/关闭
		OLED_WR_Byte(0x20,OLED_CMD); //设置内存地址模式
		OLED_WR_Byte(0x02,OLED_CMD); //[1:0],00,列地址模式;01,行地址模式;10,页地址模式;默认10;
		OLED_WR_Byte(0xA1,OLED_CMD); //段重定义设置,bit0:0,0->0;1,0->127;
		OLED_WR_Byte(0xC0,OLED_CMD); //设置COM扫描方向;bit3:0,普通模式;1,重定义模式 COM[N-1]->COM0;N:驱动路数
		OLED_WR_Byte(0xDA,OLED_CMD); //设置COM硬件引脚配置
		OLED_WR_Byte(0x12,OLED_CMD); //[5:4]配置	 
		OLED_WR_Byte(0x81,OLED_CMD); //对比度设置
		OLED_WR_Byte(0xEF,OLED_CMD); //1~255;默认0X7F (亮度设置,越大越亮)
		OLED_WR_Byte(0xD9,OLED_CMD); //设置预充电周期
		OLED_WR_Byte(0xf1,OLED_CMD); //[3:0],PHASE 1;[7:4],PHASE 2;
		OLED_WR_Byte(0xDB,OLED_CMD); //设置VCOMH 电压倍率
		OLED_WR_Byte(0x30,OLED_CMD); //[6:4] 000,0.65*vcc;001,0.77*vcc;011,0.83*vcc;
		OLED_WR_Byte(0xA4,OLED_CMD); //全局显示开启;bit0:1,开启;0,关闭;(白屏/黑屏)
		OLED_WR_Byte(0xA6,OLED_CMD); //设置显示方式;bit0:1,反相显示;0,正常显示	    						   
		OLED_WR_Byte(0xAF,OLED_CMD); //开启显示	 
}



/*更新显存到OLED*/
void OLED_Refresh_Gram(void)
{
	u8 i,n;
	for(i=0;i<8;++i)
	{
		OLED_WR_Byte(0xb0+i,OLED_CMD);
		OLED_WR_Byte(0x00,OLED_CMD);
		OLED_WR_Byte(0x10,OLED_CMD);
		for(n=0;n<128;n++)
		{
			OLED_WR_Byte(OLED_GRAM[n][i],OLED_DATA);
		}
	}
}

//开启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++)for(n=0;n<128;n++)OLED_GRAM[n][i]=0X00;  
	OLED_Refresh_Gram();//更新显示
}
//画点 
//x:0~127
//y:0~63
//t:1 填充 0,清空				   
void OLED_DrawPoint(u8 x,u8 y,u8 t)
{
	u8 pos,bx,temp=0;
	if(x>127||y>63)return;//超出范围了.
	pos=7-y/8;
	bx=y%8;
	temp=1<<(7-bx);
	if(t)OLED_GRAM[x][pos]|=temp;
	else OLED_GRAM[x][pos]&=~temp;	    
}
//x1,y1,x2,y2 填充区域的对角坐标
//确保x1<=x2;y1<=y2 0<=x1<=127 0<=y1<=63	 	 
//dot:0,清空;1,填充	  
void OLED_Fill(u8 x1,u8 y1,u8 x2,u8 y2,u8 dot)  
{  
	u8 x,y;  
	for(x=x1;x<=x2;x++)
	{
		for(y=y1;y<=y2;y++)OLED_DrawPoint(x,y,dot);
	}													    
	OLED_Refresh_Gram();//更新显示
}


/*显示字符*/
void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 size,u8 mode)
{      			    
	u8 temp,t,t1;
	u8 y0=y;
	u8 csize=(size/8+((size%8)?1:0))*(size/2);		//得到字体一个字符对应点阵集所占的字节数
	chr=chr-' ';//得到偏移后的值		 
    for(t=0;t<csize;t++)
    {   
			if(size==12)temp=lsasc2_1206[chr][t]; 	 	//调用1206字体
			else if(size==16)temp=lsasc2_1608[chr][t];	//调用1608字体
			else if(size==24)temp=lsasc2_2412[chr][t];	//调用2412字体
	//		else if(size==32)temp=lsChineseFont_16_16[43][t];  //调用汉字
			else return;								//没有的字库
			for(t1=0;t1<8;t1++)
			{
				if(temp&0x80)OLED_DrawPoint(x,y,mode);
				else OLED_DrawPoint(x,y,!mode);
				temp<<=1;
				y++;
				if((y-y0)==size)
				{
					y=y0;
					x++;
					break;
				}
			}  	 
    }          
}

u32 mypow(u8 m,u8 n)
{
	u32 result=1;	 
	while(n--)result*=m;    
	return result;
}	

void OLED_ShowNum(u8 x,u8 y,u32 num,u8 len,u8 size)
{         	
	u8 t,temp;
	u8 enshow=0;						   
	for(t=0;t<len;t++)
	{
		temp=(num/mypow(10,len-t-1))%10;
		if(enshow==0&&t<(len-1))
		{
			if(temp==0)
			{
				OLED_ShowChar(x+(size/2)*t,y,' ',size,1);
				continue;
			}else enshow=1; 
		 	 
		}
	 	OLED_ShowChar(x+(size/2)*t,y,temp+'0',size,1); 
	}
} 

void OLED_ShowString(u8 x,u8 y,const u8 *p,u8 size)
{	
    while((*p<='~')&&(*p>=' '))//判断是不是非法字符!
    {       
        if(x>(128-(size/2))){x=0;y+=size;}
        if(y>(64-size)){y=x=0;OLED_Clear();}
        OLED_ShowChar(x,y,*p,size,1);	 
        x+=size/2;
        p++;
    }  
	
}	

void OLED_ShowC(u16 x,u16 y,u16 chr,u8 size,u8 mode)
{      			    
	u8 temp,t,t1;
	u8 y0=y;
	u8 csize=(size/8+((size%8)?1:0))*(size/2);		//得到字体一个字符对应点阵集所占的字节数
	chr=chr-' ';//得到偏移后的值		 
    for(t=0;t<csize;t++)
    {   
			if(size==12)temp=lsasc2_1206[chr][t]; 	 	//调用1206字体
			else if(size==16)temp=lsasc2_1608[chr][t];	//调用1608字体
			else if(size==24)temp=lsasc2_2412[chr][t];	//调用2412字体
			//else if(size==32)temp=lsChineseFont_16_16[chr][t];  //调用汉字
			else return;								//没有的字库
			for(t1=0;t1<8;t1++)
			{
				if(temp&0x80)OLED_DrawPoint(x,y,mode);
				else OLED_DrawPoint(x,y,!mode);
				temp<<=1;
				y++;
				if((y-y0)==size)
				{
					y=y0;
					x++;
					break;
				}
			}  	 
    }          
}

/*下面是我用oled做贪吃蛇的部分代码*/
void OLED_ShowSideRectangle(void)
{ 
	
	  u8 x = 0,y=0;
		for(x = 20;x<107;x++)
		{
			for(y=0;y<8;y++)
			{
				if(y == 0)
				{
					OLED_GRAM[x][y] = 0x01;
				}
				if(y==7)
				{
					OLED_GRAM[x][y] = 0x80;
				}
				if(x == 20 || x== 106)
				{
					OLED_GRAM[x][y] = 0xff;
				}
			}
		}
}

void OLED_ShowSnake(uint16_t footx,uint16_t footy,uint8_t *flag)
{
	/*u8 test[128][64];
	test[64][32] = 1;*/
	/*static u8 a = 1;
	OLED_GRAM[64][4] = a;
	a++;*/
	u8 tmp = (footy/8);
	u8 tm = (footy%8);
	OLED_GRAM[64][tmp] = (1<<tm);
	
}