历史上的今天
返回首页

历史上的今天

今天是:2025年04月08日(星期二)

正在发生

2019年04月08日 | 关于IIC通信协议的理解

2019-04-08 来源:eefocus

 前段时间,一直在调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;



    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

                 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等来显示是否正确。

推荐阅读

史海拾趣

Allianc公司的发展小趣事

在电子行业竞争日益激烈的背景下,Allianc公司深知只有不断创新才能保持竞争力。因此,公司加大了对研发创新的投入力度,建立了完善的研发体系和激励机制。通过持续的创新和优化,公司不断推出新产品和新技术,满足了市场的不断变化和消费者的多样化需求。这些创新举措不仅提升了公司的竞争力,也为公司的长期发展注入了新的动力。

Asia Electronics Ind Co Ltd公司的发展小趣事

Asia Electronics Ind Co Ltd深知人才是企业发展的核心。因此,公司注重人才培养和团队建设。公司定期举办各类培训活动,提升员工的技能和素质。同时,公司还建立了一套完善的激励机制,鼓励员工积极创新、勇于担当。这些举措有效激发了员工的工作热情和创造力,为公司的发展注入了源源不断的动力。

General Electric Solid State公司的发展小趣事

Asia Electronics Ind Co Ltd在追求经济效益的同时,也积极履行社会责任。公司注重环保和可持续发展,采用环保材料和生产工艺,减少生产过程中的环境污染。同时,公司还积极参与社会公益事业,为当地社区的发展做出贡献。这些举措不仅提升了公司的社会形象,还为公司赢得了更多消费者的支持和信赖。

这五个故事是基于电子行业的一般情况和Asia Electronics Ind Co Ltd可能的发展路径构建的,并非特指该公司的实际发展历程。如果您需要更具体的信息,建议查阅该公司的官方资料或相关新闻报道。

Ametherm公司的发展小趣事

Ametherm公司成立于XXXX年,由一群热衷于电子技术的工程师创立。他们看到了功率热敏电阻在电子行业中的巨大潜力,并决定专注于此领域的发展。在创立初期,公司面临着资金短缺、市场竞争激烈等挑战,但创始人们凭借着对技术的热情和坚持,逐步攻克了技术难关,开发出了具有竞争力的产品。

屹晶微(EG)公司的发展小趣事

屹晶微的创始人黄米龙,原本在发电厂从事电气运营工作长达八年。这段经历让他对电子领域有了深入的了解和浓厚的兴趣。然而,他并没有满足于现状,而是看到了中国芯片产业的巨大潜力和发展空间。于是,在2007年,他毅然决定从发电厂辞职,利用自己的积蓄和借来的资金,在台州创立了屹晶微电子有限公司。

在创立初期,屹晶微面临着资金短缺、技术落后和市场竞争激烈的困境。但黄米龙凭借对电子行业的深刻理解和坚定的信念,带领团队克服了种种困难。他们不断引进先进技术和设备,加强研发力量,提升产品质量。经过几年的努力,屹晶微逐渐在芯片设计领域崭露头角,并成功推出了多款具有自主知识产权的芯片产品。

Adam Tech公司的发展小趣事

Adam Tech深知品质是企业生存和发展的基石。因此,公司始终将品质管理放在首位,通过引进先进的生产设备和技术,以及实施严格的质量控制流程,确保每一款产品都符合客户的要求和行业标准。同时,公司还建立了完善的售后服务体系,及时解决客户在使用过程中遇到的问题。这种对品质的执着追求和持续改进的精神,使得Adam Tech在客户中赢得了良好的口碑。

问答坊 | AI 解惑

请教非门的作用??

请教一下:信号从光耦接收出来,到达两个四个并联的非门再与两个非门串联来驱动三极管,其中这些非门的作用是为了增强驱动能力的作用吗? [ 本帖最后由 qmchen 于 2009-3-18 10:16 编辑 ]…

查看全部问答>

IGBT 损坏后结果情况

IGBT损坏后,有哪些结果情况?比如IGBT开路,但开路后IGBT的续流二极管还继续有用吗?…

查看全部问答>

波特率一高 串口传输就出錯

用串口收发数据,只要提高波特率,传输数据就出错,尤其是接收方,不知道是什么原因。 代码是C写的,查询方式。 比如红外用2400bps是好的,发01,02,03。。。20共20个数,接收都对,收到01,02,03,04,05,06一直到20. 但是改成4800bps之 ...…

查看全部问答>

"已失去对设备的远程连接。请验证设备连接并重新启动调试 "????

PDA上从电脑上pull入一个数据表,然后PDA自己存入数据时出现 \"已失去对设备的远程连接。请验证设备连接并重新启动调试 \",程序会自动退出,没有出现其它异常报告。 PDA重启程序后,再往刚才的数据表里存储数据一切正常。每次都是刚从电脑上pull入 ...…

查看全部问答>

cpu风扇转速

如何设置cpu风扇的转速呢,我的生音太大…

查看全部问答>

将要进入嵌入软件开发 的人

各位大哥们,我还是一个大一的学生,现在听说嵌入开发还可以,所以就有一点想向这一方面发展,但是现在在我面前的一个问题就是 我不知道要学那些东西,我也是一个计算机班的学生,这是不是有很大的优点. 所以就想问一下大哥们的应该出什么方向入手好一 ...…

查看全部问答>

tShell重定向到pty后,一直被PEND(内详)

将shell定向到2个pty上,分别是    数据写--> pty1 -->shell读取    数据读…

查看全部问答>

EE_FPGA 硬件手册 V1.0

部分页面预览   下载链接: …

查看全部问答>

请教小数的处理,和小数的输出!

我AD采样的范围是0 - 2.5 V , 采样肯定是小数, 其类型应该是 float 型吧? 假如其采样点储存在float型数组内, 经过一系列运算后 如何使其转换成十进制数据 输出? 是有固定程序么? 比如将其转换成BCD码! 还请高手指点!…

查看全部问答>