当前位置:首页 > 消费电子 > 消费电子
[导读]前段时间,一直在调SDRAM与VGA的驱动,搞了很长一段时间,参考了很多资料,最终终于思路理清了,不过鉴于手上没有相关的硬件电路,所以暂时搁置了,回归正题,先来看IIC之间

前段时间,一直在调SDRAM与VGA的驱动,搞了很长一段时间,参考了很多资料,最终终于思路理清了,不过鉴于手上没有相关的硬件电路,所以暂时搁置了,回归正题,先来看IIC之间的通信吧。

首先,IIC通信与UART,还有SPI统称为串行接口通信,不过它们之间还是有区别的,如UART的负电平逻辑,还有UART通信不需要时钟,只需要特定的波特率即可,SPI与IIC都可以有一个主机,多个从机的情况,不过IIC适用于短距离传输,如片间通信,摄像头的配置等场景。

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

 

 

如图所示,我们知道IIC一个主机可以悬挂多个从机,所以地址线A2,A1,A0 可以实行片选的功能,那么WP这个引脚的功能就是当WP悬空或者接地的时候,表示这时的EEPROM既可以读,也可以写,当WP接电源时,则只可以读而不能写。

SCL与SDL这两个引脚,必须上拉,否则驱动能力不够,无法进行正常的IIC通信。

OK,硬件接口已经介绍清楚了,那么我们现在开始来看协议了。

首先IIC分为字节读写和页面读写,首先来看字节读写的协议:

 

 

如上图所示,如果我们要向EEPROM中写入一个字节的数据,得有如下几个步骤:

1.开始信号——在SCLK的高电平器件,拉低SDA的信号(由1 变为0)。

2.控制字节——即器件地址,就是你操作那一块EEPROM。

3.ACK信号——由从机发出,主机为接收,所以在此阶段,sda_link必须置为0,即为读取这个应答信号,所以在SCLK的高点平期间。

4.字节地址——即某一块EEPROM里面的哪一个地址。

5.ACK信号——与上述相同。

6.数据信号——即你往某个地址里面写入的8位数据。

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

8.结束信号——在SCLK的高电平期间,拉高SDA信号,表示通信结束。

再来看读的时序:

 

 

由上图可看出读时序的前面处理方式与写相同,不同的时在第三个ACK信号来了之后,如果是读,那么会又有一个起始信号,紧接着读器件地址,然后应答,再然后读数据,再然后在SCLK的低电平期间发送一个NO ACK信号,要记住这个信号由主机发出,然后紧接着一个结束信号。

由上述读写时序我们可知,通信的起始均在SCLK的高电平期间发生跳变,这就据定了我们其他信号跳变均在SCLK的下降沿,SCLK高电平期间数据稳定,适用于读(即低电平改变数据,高电平采集数据)。

具体过程如下:

首先板子上电来个初始化需要来个延时,具体多少用计数器自己搞定。

代码如下:

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而我们的系统时钟为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高电平期间采集数据,低电平期间改变数据,那么当然,这个“期间”肯定时时钟沿中间最好啦,毕竟更容易满足建立时间与保持时间,很稳定的。

具体代码如下:

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,读写定义了那么多个过程,当然需要状态机来搞定啦,定义变量如下:

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,再来个宏定义,假设写入是这几个地址,这几个数据。

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;[!--empirenews.page--]

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)

// 你要是不等它松开才恢复初始状态,那么你一旦恢复初始状态SW1_r就为低电平,他又开始写了,所以为了避免重复写入数据。

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,搞定,输出当然可以连接数码管,连接LED等来显示是否正确。

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭