OLED12864_I2C接口使用教程_51單片機(jī) 現(xiàn)在能買到的OLED12864顯示屏大多為SPI和I2C接口的,I2C通信協(xié)議只需要兩條總線就可以進(jìn)行通信,下面介紹一下如何用51單片機(jī)使用I2C接口的OLED12864。
首先介紹一下I2C通信協(xié)議,I2C(Inter-Integrated Circuit)字面上的意思是集成電路之間,它其實(shí)是I2CBus簡稱,所以中文應(yīng)該叫集成電路總線,它是一種串行通信總線,使用多主從架構(gòu),由飛利浦公司在1980年代為了讓主板、嵌入式系統(tǒng)或手機(jī)用以連接低速周邊設(shè)備而發(fā)展。I2C的正確讀法為“I平方C”("I-squared-C")。 I2C只使用兩條雙向漏極開路(Open Drain)(串行數(shù)據(jù)(SDA)及串行時(shí)鐘頻率(SCL))總線,且利用上拉電阻將兩條總線的電位上拉。I2C允許相當(dāng)大的工作電壓范圍,但典型的電壓準(zhǔn)位為+3.3V或+5V。 I2C的參考設(shè)計(jì)使用一個(gè)7比特長度的地址空間但保留了16個(gè)地址,所以在一組總線最多可和112個(gè)節(jié)點(diǎn)通信[a]。常見的I2C總線依傳輸速率的不同而有不同的模式:標(biāo)準(zhǔn)模式(100 Kbit/s)、低速模式(10 Kbit/s),但時(shí)鐘頻率可被允許下降至零,這代表可以暫停通信。而新一代的I2C總線可以和更多的節(jié)點(diǎn)(支持10比特長度的地址空間)以更快的速率通信:快速模式(400 Kbit/s)、高速模式(3.4 Mbit/s)。 對(duì)于I2C通信協(xié)議,需要補(bǔ)充的一點(diǎn)是:在實(shí)際通信傳輸數(shù)據(jù)時(shí),SCL總線拉高的時(shí)間只要大于1.5μs都能夠正常傳輸數(shù)據(jù)。 OLED12864的裸屏是由SSD1306驅(qū)動(dòng)的,I2C接口的OLED12864模塊對(duì)外一共有4個(gè)接口,從左到右分別是GND(接地)、VCC(電源正極,可加3.3V,也可加5V)、SCL(時(shí)鐘總線)、SDA(數(shù)據(jù)總線):
Oled_12864_1.jpg (988.26 KB, 下載次數(shù): 161)
下載附件
Oled_12864_1
2018-3-24 13:07 上傳
模塊背面的IIC ADRESSSELECT表示該模塊在I2C通信作為從機(jī)時(shí)的地址,當(dāng)中間的腳用電阻和左邊接起來時(shí),地址為0x78,當(dāng)和右邊接起來時(shí),地址為0x7A。
Oled_12864_2.jpg (1.23 MB, 下載次數(shù): 215)
下載附件
Oled_12864_2
2018-3-24 13:08 上傳
SSD1306的I2C總線數(shù)據(jù)格式,可以看出,往OLED12864寫數(shù)據(jù)時(shí),先發(fā)送一個(gè)起始信號(hào),接著發(fā)送從機(jī)地址,從機(jī)地址帶有讀寫位(低電平為寫),之后就可以發(fā)送指令或數(shù)據(jù)。在發(fā)送指令或數(shù)據(jù)之前,一般都需要發(fā)送一個(gè)控制字節(jié),如圖,控制字節(jié)的最高位為連續(xù)位(如果連續(xù)位為0,接下來發(fā)送的信息只能包含數(shù)據(jù)字節(jié)),次高位為數(shù)據(jù)/指令選擇位(該位聲明接下來發(fā)送的是數(shù)據(jù)還是指令,0為指令,1為數(shù)據(jù)),控制字節(jié)的低六位為0?梢栽谝粋(gè)聲明連續(xù)發(fā)送數(shù)據(jù)的控制字節(jié)后面跟上多個(gè)數(shù)據(jù)字節(jié)。
Data Format.png (52.88 KB, 下載次數(shù): 174)
下載附件
2018-3-24 13:15 上傳
作者:Ace 轉(zhuǎn)載請注明作者,謝謝!
0.png (47.84 KB, 下載次數(shù): 190)
下載附件
2018-3-24 19:11 上傳
發(fā)現(xiàn)直接貼代碼有問題,,所以就把源代碼作為附件傳上來。
Oled12864_SRC.rar
(39.86 KB, 下載次數(shù): 869)
2018-3-24 13:56 上傳
點(diǎn)擊文件名下載附件
下載積分: 黑幣 -5
注釋詳細(xì)的51單片機(jī)源程序如下(IIC.c):
- #include "IIC.h"
- void delay5us()
- {
- }
- void I2C_init() //初始化
- {
- SDA = 1;
- _nop_();
- SCL = 1;
- _nop_(); //空閑時(shí),兩條線均為高電平
- }
- /*I2C通信起始信號(hào)*/
- void I2C_start()
- {
- SCL = 1; //此時(shí)主機(jī)有總線控制權(quán),先把SCL線拉高
- _nop_(); //穩(wěn)定一下
- SDA = 1; //把SDA線拉高,以便發(fā)出起始信號(hào)(不確定是否為高)
- delay5us();//通信協(xié)議規(guī)定延時(shí)大于4.7us
- SDA = 0; //拉低SDA線,制造下降沿的起始信號(hào)
- delay5us();//通信協(xié)議規(guī)定延時(shí)大于4us
- }
-
- /*I2C通信終止信號(hào)*/
- void I2C_stop()
- {
- SDA = 0; //拉低SDA線,以便發(fā)出終止信號(hào)
- _nop_(); //穩(wěn)定一下
- SCL = 1; //拉高SCL線
- delay5us();//通信協(xié)議規(guī)定延時(shí)大于4us
- SDA = 1; //拉高SDA線
- delay5us();//通信協(xié)議規(guī)定延時(shí)大于4.7us
- }
- /*從機(jī)應(yīng)答檢測*/
- bit Test_ack()
- {
- SCL = 1;
- //拉高SCL線,以便主機(jī)讀取從機(jī)發(fā)出的應(yīng)答或非應(yīng)答信號(hào)
- delay5us();//通信協(xié)議規(guī)定延時(shí)大于4us
- if(SDA)
- //主機(jī)讀取的SDA線為高,說明從機(jī)發(fā)送了一個(gè)非應(yīng)答信號(hào)
- {
- SCL = 0;/*接下來準(zhǔn)備發(fā)送停止信號(hào),所以讓時(shí)鐘總線SCL拉低,
- 讓I2C_stop();函數(shù)中的SDA可變?yōu)?*/
- _nop_();//穩(wěn)定總線
- I2C_stop();
- return 0;//結(jié)束檢測從機(jī)應(yīng)答函數(shù)
- }
- else
- //主機(jī)讀取的SDA線為低,說明從機(jī)發(fā)送了一個(gè)應(yīng)答信號(hào)
- {
- SCL = 0;/*將時(shí)鐘總線SCL拉低,此時(shí)SDA上數(shù)據(jù)的變化才有效,
- 因?yàn)榻酉聛頃?huì)繼續(xù)發(fā)數(shù)據(jù)*/
- _nop_();//穩(wěn)定總線
- return 1;
- }
- }
- /*I2C通信:發(fā)送一個(gè)字節(jié)*/
- void I2C_send_byte(uint8_t byte)
- {
- uint8_t i;//聲明一個(gè)計(jì)數(shù)變量i
- for(i = 0; i < 8; i++)//一個(gè)字節(jié)有8位,發(fā)8次
- {
- SCL = 0;//拉低SCL,讓數(shù)據(jù)總線SDA變化有效
- _nop_();//穩(wěn)定總線
- if(byte & 0x80)//1000 0000 & byte
- /*如果(要發(fā)送數(shù)據(jù)的當(dāng)前傳輸位)byte的最高位為1,執(zhí)行該if語句,
- 如果(要發(fā)送數(shù)據(jù)的當(dāng)前傳輸位)byte的最高位為0,不執(zhí)行該if語句*/
- {
- SDA = 1;
- //(當(dāng)前傳輸位)byte的最高位為1,即把1放到SDA線上
- _nop_();//穩(wěn)定總線
- }
- else
- //如果(當(dāng)前發(fā)送位)byte的最高位為0(不為1),給SDA送0
- {
- SDA = 0;
- _nop_();//穩(wěn)定總線
- }
- SCL = 1;//拉高SCL線,使從機(jī)能夠從SDA線上讀取到當(dāng)前的數(shù)據(jù)
- _nop_();//穩(wěn)定總線
- byte <<= 1;
- /*使byte左移一位,即原來已被發(fā)送到SDA線上的最高位被移除,
- 第七位(還未發(fā)送的數(shù)據(jù)位)變成最高位變?yōu)橄乱淮窝h(huán)的當(dāng)前發(fā)送位*/
- }
- SCL = 0;
- //發(fā)送完數(shù)據(jù)之后,將SCL拉低,以便從機(jī)改變SDA線,發(fā)出應(yīng)答信號(hào)
- _nop_();//穩(wěn)定總線
- SDA = 1;//釋放總線控制權(quán)
- _nop_();//穩(wěn)定總線
- }
復(fù)制代碼
OLED_12864.c
- #include "OLED_12864.h"
- void Delay300ms()
- {
- }
- /*寫指令函數(shù),第一個(gè)參數(shù)為指令,第二、三個(gè)參數(shù)選擇是否需要通信開始和結(jié)束函數(shù),=1有,=0沒有*/
- bit OLED12864_Write_Commmand(uint8_t cmd, bit start, bit stop)
- {
- if(start)
- {
- I2C_start();
- I2C_send_byte(OLED_12864_Address+0);//寫從機(jī)地址,并且加上讀寫標(biāo)志位(最后一位)
- if(!Test_ack())
- {
- return 0;
- }
- /*執(zhí)行從機(jī)應(yīng)答檢測函數(shù),如果從機(jī)發(fā)送了非應(yīng)答信號(hào),那么就退出數(shù)據(jù)發(fā)送函數(shù)*/
- }
- I2C_send_byte(0x80 | 0x00 | 0x00); //Co位為1(接下來要傳指令),DC為0(接下來是指令)
- if(!Test_ack())
- {
- return 0;
- }
- /*執(zhí)行從機(jī)應(yīng)答檢測函數(shù),如果從機(jī)發(fā)送了非應(yīng)答信號(hào),那么就退出數(shù)據(jù)發(fā)送函數(shù)*/
- I2C_send_byte(cmd);
- if(!Test_ack())
- {
- return 0;
- }
- /*執(zhí)行從機(jī)應(yīng)答檢測函數(shù),如果從機(jī)發(fā)送了非應(yīng)答信號(hào),那么就退出數(shù)據(jù)發(fā)送函數(shù)*/
- if(stop)
- I2C_stop();
- return 1;
- }
- /*寫連續(xù)數(shù)據(jù)函數(shù),第一個(gè)參數(shù)為數(shù)據(jù),第二個(gè)參數(shù)為發(fā)送連多少位連續(xù)的數(shù)據(jù),第三、四個(gè)參數(shù)選擇是否需要通信開始和結(jié)束函數(shù),=1有,=0沒有*/
- bit OLED12864_Write_Continuous_Data(uint8_t* dat, uint8_t count, bit start, bit stop)
- {
- uint8_t i = 0;//定義計(jì)數(shù)變量
- if(start)
- {
- I2C_start();
- I2C_send_byte(OLED_12864_Address+0);//寫從機(jī)地址,并且加上讀寫標(biāo)志位(最后一位)
- if(!Test_ack())
- {
- return 0;
- }
- /*執(zhí)行從機(jī)應(yīng)答檢測函數(shù),如果從機(jī)發(fā)送了非應(yīng)答信號(hào),那么就退出數(shù)據(jù)發(fā)送函數(shù)*/
- }
- I2C_send_byte(0x00 | 0x40 | 0x00); //Co位為0(接下來只傳數(shù)據(jù)),DC為1(接下來是數(shù)據(jù))
- if(!Test_ack())
- {
- return 0;
- }
- /*執(zhí)行從機(jī)應(yīng)答檢測函數(shù),如果從機(jī)發(fā)送了非應(yīng)答信號(hào),那么就退出數(shù)據(jù)發(fā)送函數(shù)*/
- for(i = 0 ;i < count ;i++)
- {
- I2C_send_byte(*dat++);
- if(!Test_ack())
- {
- return 0;
- }
- /*執(zhí)行從機(jī)應(yīng)答檢測函數(shù),如果從機(jī)發(fā)送了非應(yīng)答信號(hào),那么就退出數(shù)據(jù)發(fā)送函數(shù)*/
- }
- if(stop)
- I2C_stop();
- return 1;
- }
- /*寫相同的連續(xù)數(shù)據(jù)函數(shù),第一個(gè)參數(shù)為數(shù)據(jù),第二參數(shù)為發(fā)送的次數(shù)*/
- bit OLED12864_Write_Same_Continuous_Data(uint8_t dat, uint8_t count)
- {
- uint8_t i = 0;//定義計(jì)數(shù)變量
- I2C_start();
- I2C_send_byte(OLED_12864_Address+0);//寫從機(jī)地址,并且加上讀寫標(biāo)志位(最后一位)
- if(!Test_ack())
- {
- return 0;
- }
- /*執(zhí)行從機(jī)應(yīng)答檢測函數(shù),如果從機(jī)發(fā)送了非應(yīng)答信號(hào),那么就退出數(shù)據(jù)發(fā)送函數(shù)*/
- I2C_send_byte(0x00 | 0x40 | 0x00); //Co位為0(接下來只傳數(shù)據(jù)),DC為1(接下來是數(shù)據(jù))
- if(!Test_ack())
- {
- return 0;
- }
- /*執(zhí)行從機(jī)應(yīng)答檢測函數(shù),如果從機(jī)發(fā)送了非應(yīng)答信號(hào),那么就退出數(shù)據(jù)發(fā)送函數(shù)*/
- for(i = 0 ;i < count ;i++ )
- {
- I2C_send_byte(dat);
- if(!Test_ack())
- {
- return 0;
- }
- /*執(zhí)行從機(jī)應(yīng)答檢測函數(shù),如果從機(jī)發(fā)送了非應(yīng)答信號(hào),那么就退出數(shù)據(jù)發(fā)送函數(shù)*/
- }
-
- I2C_stop();
- return 1;
- }
-
- //Set the cursor position of start
- void OLED12864_SetPos(uint8_t x, uint8_t y)
- {
- OLED12864_Write_Commmand(0xb0+y,1,0);
- OLED12864_Write_Commmand(((x&0xf0)>>4)|0x10,0,0);
- OLED12864_Write_Commmand((x&0x0f)|0x01,0,1);
- }
-
- //Fill screen wit data
- //0x00 is black
- //0xff is blue
- void OLED12864_Fill(uint8_t fill_Data)
- {
- uint8_t m;
- //uint8_t n;
- for(m=0;m<8;m++)
- {
- OLED12864_Write_Commmand(0xb0+m,1,0); //page0-page1
- OLED12864_Write_Commmand(0x00,0,0); //low column start address
- OLED12864_Write_Commmand(0x10,0,1); //high column start address
- OLED12864_Write_Same_Continuous_Data(fill_Data, 128);
- }
- }
-
- void OLED12864_CLS_LINE(uint8_t rowIndex)
- {
- uint8_t n=0;
- OLED12864_Write_Commmand(0xb0+rowIndex,1,0); //page0-page1
- OLED12864_Write_Commmand(0x00,0,0); //low column start address
- OLED12864_Write_Commmand(0x10,0,1); //high column start address
- OLED12864_Write_Same_Continuous_Data(0x00, 128);
- }
-
- //clear screen( fill screen with black)
- void OLED12864_CLS(void)
- {
- OLED12864_Fill(0x00);
- }
-
- //--------------------------------------------------------------
- // wake up screen from hibernation
- //--------------------------------------------------------------
- void OLED12864_ON(void)
- {
- OLED12864_Write_Commmand(0X8D,1,0); //set charge
- OLED12864_Write_Commmand(0X14,0,0); //open charge
- OLED12864_Write_Commmand(0XAF,0,1); //OLED wake up
- }
-
- //--------------------------------------------------------------
- // Prototype : void OLED12864_OFF(void)
- // Calls :
- // Parameters : none
- // Description : ?OLED?? -- ?????,OLED????10uA
- //--------------------------------------------------------------
- void OLED12864_OFF(void)
- {
- OLED12864_Write_Commmand(0X8D,1,0); //set charge
- OLED12864_Write_Commmand(0X10,0,0); //close charge
- OLED12864_Write_Commmand(0XAE,0,1); //OLED hibernate
- }
- bit OLED12864_Initial()
- {
- Delay300ms(); // very important delay
-
- OLED12864_Write_Commmand(0xAE,1,0); //display off
- OLED12864_Write_Commmand(0x20,0,0); //Set Memory Addressing Mode
- OLED12864_Write_Commmand(0x10,0,0); //00,Horizontal Addressing Mode;01,Vertical Addressing Mode;10,Page Addressing Mode (RESET);11,Invalid
- OLED12864_Write_Commmand(0xb0,0,0); //Set Page Start Address for Page Addressing Mode,0-7
- OLED12864_Write_Commmand(0xc8,0,0); //Set COM Output Scan Direction
- OLED12864_Write_Commmand(0x00,0,0); //---set low column address
- OLED12864_Write_Commmand(0x10,0,0); //---set high column address
- OLED12864_Write_Commmand(0x40,0,0); //--set start line address
- OLED12864_Write_Commmand(0x81,0,0); //--set contrast control register
- OLED12864_Write_Commmand(0xff,0,0); //???? 0x00~0xff
- OLED12864_Write_Commmand(0xa1,0,0); //--set segment re-map 0 to 127
- OLED12864_Write_Commmand(0xa6,0,0); //--set normal display
- OLED12864_Write_Commmand(0xa8,0,0); //--set multiplex ratio(1 to 64)
- OLED12864_Write_Commmand(0x3F,0,0); //
- OLED12864_Write_Commmand(0xa4,0,0); //0xa4,Output follows RAM content;0xa5,Output ignores RAM content
- OLED12864_Write_Commmand(0xd3,0,0); //-set display offset
- OLED12864_Write_Commmand(0x00,0,0); //-not offset
- OLED12864_Write_Commmand(0xd5,0,0); //--set display clock divide ratio/oscillator frequency
- OLED12864_Write_Commmand(0xf0,0,0); //--set divide ratio
- OLED12864_Write_Commmand(0xd9,0,0); //--set pre-charge period
- OLED12864_Write_Commmand(0x22,0,0); //
- OLED12864_Write_Commmand(0xda,0,0); //--set com pins hardware configuration
- OLED12864_Write_Commmand(0x12,0,0);
- OLED12864_Write_Commmand(0xdb,0,0); //--set vcomh
- OLED12864_Write_Commmand(0x20,0,0); //0x20,0.77xVcc
- OLED12864_Write_Commmand(0x8d,0,0); //--set DC-DC enable
- OLED12864_Write_Commmand(0x14,0,0); //
- OLED12864_Write_Commmand(0xaf,0,1); //--turn on oled panel
-
- OLED12864_CLS();
- return 1;
- }
-
- //--------------------------------------------------------------
- // show string
- // x,y -- start position(x:0~127,column; y:0~7,row);
- // ch[] -- the string to show;
- // TextSize -- (1:6*8 ; 2:8*16)
- //OLED12864_ShowStr(0,3,"I2C Test",1);// 6*8
- //OLED12864_ShowStr(0,4,"Hello Delta",2) //8*16
- //--------------------------------------------------------------
- void OLED12864_ShowStr(uint8_t x, uint8_t y, uint8_t ch[], uint8_t TextSize)
- {
- uint8_t c = 0,i = 0,j = 0;
- switch(TextSize)
- {
- case 1:
- {
- while(ch[j] != '\0' && ch[j] !='\n')
- {
- c = ch[j] - 32;
- if(x > 126)
- {
- x = 0;
- y++;
- }
- OLED12864_SetPos(x,y);
- /*for(i=0;i<6;i++)
- OLED12864_Write_Data(F6x8[c][i],1,1);*/
- OLED12864_Write_Continuous_Data(&F6x8[c][0],6,1,1);//將該位的指針傳給函數(shù),并且寫連續(xù)的6個(gè)數(shù)據(jù)即可
- x += 6;
- j++;
- }
- }break;
- case 2:
- {
- while(ch[j] != '\0' && ch[j] !='\n')
- {
- c = ch[j] - 32;
- if(x > 120)
- {
- x = 0;
- y++;
- }
- OLED12864_SetPos(x,y);
- /*for(i=0;i<8;i++)
- OLED12864_Write_Data(F8X16[c*16+i],1,1);
- OLED12864_SetPos(x,y+1);
- for(i=0;i<8;i++)
- OLED12864_Write_Data(F8X16[c*16+i+8],1,1);*/
- OLED12864_Write_Continuous_Data(&F8X16[c*16],8,1,1);//將該位的指針傳給函數(shù),并且寫連續(xù)的8個(gè)數(shù)據(jù)即可
- OLED12864_SetPos(x,y+1);
- OLED12864_Write_Continuous_Data(&F8X16[c*16+8],8,1,1);//將該位的指針傳給函數(shù),并且寫連續(xù)的8個(gè)數(shù)據(jù)即可
- x += 8;
- j++;
- }
- }break;
- }
- }
- uint8_t rowNum = 0;
-
- //--------------------------------------------------------------
- // x,y -- start position(x:0~127,column; y:0~7,row);
- // the N is the chinese index of F16x16 in codetab
- /*
- for(i=0;i<5;i++)
- {
- OLED12864_ShowCN(22+i*16,0,i);
- }
- ……………………
- …………限于本文篇幅 余下代碼請從51黑下載附件…………
復(fù)制代碼
|