日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當前位置:首頁 > 嵌入式 > 嵌入式硬件
[導讀]IIC協(xié)議是二線制,信號線包含SDA和SCL,且信號線是雙向的,開路結構,需要通過上拉電阻到VCC,具體的電阻值影響的是信號反應速度和驅動能力。

IIC協(xié)議是二線制,信號線包含SDA和SCL,且信號線是雙向的,開路結構,需要通過上拉電阻到VCC,具體的電阻值影響的是信號反應速度和驅動能力。

首先,IIC通信與UART,還有SPI統(tǒng)稱為串行接口通信,不過它們之間還是有區(qū)別的,如UART的負電平邏輯,還有UART通信不需要時鐘,只需要特定的波特率即可,SPI與IIC都可以有一個主機,多個從機的情況,不過IIC適用于短距離傳輸,如片間通信,攝像頭的配置等場景。

要搞定IIC首先來看IIC的硬件接口:

 

 

如圖所示,我們知道IIC一個主機可以懸掛多個從機,所以地址線A2,A1,A0 可以實行片選的功能,那么WP這個引腳的功能就是當WP懸空或者接地的時候,表示這時的EEPROM既可以讀,也可以寫,當WP接電源時,則只可以讀而不能寫。

SCL與SDL這兩個引腳,必須上拉,否則驅動能力不夠,無法進行正常的IIC通信。

OK,硬件接口已經(jīng)介紹清楚了,那么我們現(xiàn)在開始來看協(xié)議了。

首先IIC分為字節(jié)讀寫和頁面讀寫,首先來看字節(jié)讀寫的協(xié)議:

 

 

如上圖所示,如果我們要向EEPROM中寫入一個字節(jié)的數(shù)據(jù),得有如下幾個步驟:

1.開始信號——在SCLK的高電平器件,拉低SDA的信號(由1 變?yōu)?)。

2.控制字節(jié)——即器件地址,就是你操作那一塊EEPROM。

3.ACK信號——由從機發(fā)出,主機為接收,所以在此階段,sda_link必須置為0,即為讀取這個應答信號,所以在SCLK的高點平期間。

4.字節(jié)地址——即某一塊EEPROM里面的哪一個地址。

5.ACK信號——與上述相同。

6.數(shù)據(jù)信號——即你往某個地址里面寫入的8位數(shù)據(jù)。

7.ACK信號——上述相同。

8.結束信號——在SCLK的高電平期間,拉高SDA信號,表示通信結束。

再來看讀的時序:

 

 

由上圖可看出讀時序的前面處理方式與寫相同,不同的時在第三個ACK信號來了之后,如果是讀,那么會又有一個起始信號,緊接著讀器件地址,然后應答,再然后讀數(shù)據(jù),再然后在SCLK的低電平期間發(fā)送一個NO ACK信號,要記住這個信號由主機發(fā)出,然后緊接著一個結束信號。

由上述讀寫時序我們可知,通信的起始均在SCLK的高電平期間發(fā)生跳變,這就據(jù)定了我們其他信號跳變均在SCLK的下降沿,SCLK高電平期間數(shù)據(jù)穩(wěn)定,適用于讀(即低電平改變數(shù)據(jù),高電平采集數(shù)據(jù))。

具體過程如下:

首先板子上電來個初始化需要來個延時,具體多少用計數(shù)器自己搞定。

代碼如下:

reg [6:0] hadware_initial_delay;

wire hadware_initial_delay_done;

always@(posedge clk or negedge rst_n)

if(!rst_n)

hadware_initial_delay<=7’d0;

else

if(hadware_initial_delay<=7’d49)

hadware_initial_delay<=hadware_initial_delay+1;else

hadware_initial_delay<=hadware_initial_delay;assign hadware_initial_delay_done=(hadware_initial_delay==7’d50)?1’b1:1’b0;OK,我們要知道IIC的速率一般就幾百KH而我們的系統(tǒng)時鐘為50M,所以需要分頻:

代碼如下:

reg [8:0] sclk_cnt;

always@(posedge clk or negedge rst_n)

if(!rst_n)

sclk_cnt<=9’d0;

else

if(hadware_initial_delay_done)

begin

if(sclk_cnt<9’d499)

sclk_cnt<=sclk_cnt+1;

else

sclk_cnt<=0;

end

assign sclk=(sclk_cnt<=9’d249)?1’b1:1’b0;OK,我們知道SCLK高電平期間采集數(shù)據(jù),低電平期間改變數(shù)據(jù),那么當然,這個“期間”肯定時時鐘沿中間最好啦,畢竟更容易滿足建立時間與保持時間,很穩(wěn)定的。

具體代碼如下:

wire sclk_posedge_middle=(sclk_cnt==9’d124)?1’b1:1’b0;wire sclk_negedge_middle=(sclk_cnt==9’d374)?1’b1:1’b0;OK,讀寫定義了那么多個過程,當然需要狀態(tài)機來搞定啦,定義變量如下:

parameter IDLE = 4’d0 ;

parameter START1 = 4’d1 ;

parameter ADD1 = 4’d2 ;

parameter ACK1 = 4’d3 ;

parameter ADD2 = 4’d4 ;

parameter ACK2 = 4’d5 ;

parameter DATA = 4’d6 ;

parameter ACK3 = 4’d7 ;

parameter STOP1 = 4’d8 ;

parameter START2 = 4’d9 ;

parameter ADD3 = 4’d10;

parameter ACK4 = 4’d11;

parameter DATA_READ = 4’d12;

parameter NO_ACK = 4’d13;

parameter STOP2 = 4’d14;

OK,再來個宏定義,假設寫入是這幾個地址,這幾個數(shù)據(jù)。

define DEVICE_READ 8'b1010_0001

define DEVICE_WRITE 8’b1010_0000

define WRITE_DATA 8'b0001_0001

define BYTE_ADDR 8’b0000_0011

SDA雙向端口,這個記住,一般這樣搞;

reg sda_link;

reg sda_out_r;

assign sda=sda_link?sda_out_r:1’bz;

當作為輸出時,對吧,使sda_link拉高,作為輸入時,輸入高阻。

各過程如下:

reg [3:0] current_state;

//reg [3:0] next_state;

reg [7:0] db_r;

reg [3:0] num;

reg [7:0] data_out_reg;

always@(posedge clk or negedge rst_n)

if(!rst_n)

begin

sda_link<=0;

db_r<=0;

num<=0;

current_state<=IDLE;

sda_out_r<=0;

data_out_reg<=8’b0;

end

else

begin

case(current_state)

IDLE:begin

sda_out_r<=1;

sda_link<=1;

if(!sw1_r||!sw2_r)

current_state<=START1;

else

current_state<=IDLE;

end

START1:if(sclk_posedge_middle)

begin

sda_out_r<=0;

db_r<=`DEVICE_WRITE;

current_state<=ADD1;

end

else

current_state<=START1;

ADD1 :

if(sclk_negedge_middle)

begin

if(num==4'd8)

begin

sda_link<=0;

num<=0;

current_state<=ACK1;

sda_out_r<=1;

end

else

begin

current_state<=ADD1;

sda_out_r<=db_r[7-num];

num<=num+1;

end

end

else

current_state<=ADD1;

ACK1:

if(sclk_posedge_middle)

// begin

// if(!sda)

// begin

begin // */current_state<=ADD2;

db_r<=`BYTE_ADDR;

end

else

current_state<=ACK1;

ADD2:begin

sda_link<=1;

if(sclk_negedge_middle)begin

if(num==4'd8)

begin

sda_link<=0;

current_state<=ACK2;

num<=4'd0;

sda_out_r<=1;

end

else

begin

num<=num+1;

current_state<=ADD2;

sda_out_r<=db_r[7-num];

end

end

else

current_state<=ADD2;

end

ACK2:

if(sclk_posedge_middle)

////begin

//if(!sda)

begin

begin

if(!sw1_r)

begin

db_r<=`WRITE_DATA;

current_state<=DATA;

end

else

if(!sw2_r)

begin

current_state<=START2;

sda_out_r<=1;

end

end

else

current_state<=ACK2;

DATA: begin

sda_link<=1;

if(sclk_negedge_middle)

begin

if(num==4'd8)

begin

num<=4'd0;

current_state<=ACK3;

sda_out_r<=1;

sda_link<=0;

end

else

begin

num<=num+1;

current_state<=DATA;

sda_out_r<=db_r[7-num];

end

end

else

current_state<=DATA;

end

ACK3: if(sclk_posedge_middle)

// begin

// if(!sda)

current_state<=STOP1;

// end

STOP1:

begin

sda_link<=1;

sda_out_r<=0;

if(sclk_posedge_middle)

begin

sda_out_r<=1;

if(sw1_r)

// 你要是不等它松開才恢復初始狀態(tài),那么你一旦恢復初始狀態(tài)SW1_r就為低電平,他又開始寫了,所以為了避免重復寫入數(shù)據(jù)。

current_state<=IDLE;

else

current_state<=STOP1;

end

else

current_state<=STOP1;

end

START2:begin

sda_link<=1;

if(sclk_posedge_middle)

begin

sda_out_r<=0;

sda_link<=1;

db_r<=`DEVICE_READ;

current_state<=ADD3 ;

end

end

ADD3: begin

if(sclk_negedge_middle)

begin

if(num==4'd8)

begin

num<=0;

sda_link<=0;

sda_out_r<=1;

current_state<=ACK4;

end

else

begin

num<=num+1;

sda_out_r<=db_r[7-num];

current_state<=ADD3;

end

end

else

current_state<=ADD3;

end

ACK4:

if(sclk_posedge_middle)

// begin

// if(!sda)

current_state<=DATA_READ;

else

current_state<=ACK4;

// end

DATA_READ:

begin

sda_link<=0;

if(sclk_posedge_middle)

begin

if(num==4'd8)

begin

sda_link<=1;

sda_out_r<=1;

current_state<=NO_ACK;

num<=4'd0;

end

else

begin

num<=num+1;

current_state<=DATA_READ;

data_out_reg[7-num]<=sda;

end

end

end

NO_ACK:

if(sclk_negedge_middle)

begin

sda_out_r<=1;

current_state<=STOP2;

end

else

current_state<=NO_ACK;

STOP2:begin

sda_out_r<=0;

sda_link<=1;

if(sclk_posedge_middle)

begin

sda_out_r<=1;

current_state<=IDLE;

end

else

current_state<=STOP2;

end

default:current_state<=IDLE;

endcase

end

assign data_out=data_out_reg;

endmodule

仿真結果如下:

 

 

OK,搞定,輸出當然可以連接數(shù)碼管,連接LED等來顯示是否正確。

本站聲明: 本文章由作者或相關機構授權發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

LED驅動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關鍵字: 驅動電源

在工業(yè)自動化蓬勃發(fā)展的當下,工業(yè)電機作為核心動力設備,其驅動電源的性能直接關系到整個系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動勢抑制與過流保護是驅動電源設計中至關重要的兩個環(huán)節(jié),集成化方案的設計成為提升電機驅動性能的關鍵。

關鍵字: 工業(yè)電機 驅動電源

LED 驅動電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個照明設備的使用壽命。然而,在實際應用中,LED 驅動電源易損壞的問題卻十分常見,不僅增加了維護成本,還影響了用戶體驗。要解決這一問題,需從設計、生...

關鍵字: 驅動電源 照明系統(tǒng) 散熱

根據(jù)LED驅動電源的公式,電感內(nèi)電流波動大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關鍵字: LED 設計 驅動電源

電動汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產(chǎn)業(yè)的重要發(fā)展方向。電動汽車的核心技術之一是電機驅動控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機驅動系統(tǒng)中的關鍵元件,其性能直接影響到電動汽車的動力性能和...

關鍵字: 電動汽車 新能源 驅動電源

在現(xiàn)代城市建設中,街道及停車場照明作為基礎設施的重要組成部分,其質量和效率直接關系到城市的公共安全、居民生活質量和能源利用效率。隨著科技的進步,高亮度白光發(fā)光二極管(LED)因其獨特的優(yōu)勢逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關鍵字: 發(fā)光二極管 驅動電源 LED

LED通用照明設計工程師會遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關鍵字: LED 驅動電源 功率因數(shù)校正

在LED照明技術日益普及的今天,LED驅動電源的電磁干擾(EMI)問題成為了一個不可忽視的挑戰(zhàn)。電磁干擾不僅會影響LED燈具的正常工作,還可能對周圍電子設備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來解決L...

關鍵字: LED照明技術 電磁干擾 驅動電源

開關電源具有效率高的特性,而且開關電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機重量也有所下降,所以,現(xiàn)在的LED驅動電源

關鍵字: LED 驅動電源 開關電源

LED驅動電源是把電源供應轉換為特定的電壓電流以驅動LED發(fā)光的電壓轉換器,通常情況下:LED驅動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關鍵字: LED 隧道燈 驅動電源
關閉