我的32單片機因為型号問題,和網絡上的很多型号都有些不同,是以需要自己編寫iic的讀寫資料、讀寫指令的代碼來操作oled,是以下面的代碼可以分為兩部分,一部分是iic的讀寫,讀寫我是根據老師給的代碼進行修改的,而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);
}