iic通信协议是什么[通俗易懂]

iic通信协议是什么[通俗易懂] iic通信协议是什么  IIC协议是二线制,信号线包含SDA和SCL,且信号线是双向的,开路结构,需要通过上拉电阻到VCC,具体的电阻值影响的是信号反应速度和驱动能力。  首先,IIC通信与UART,还有SPI统称为串行接口通信,不过它们之间还是有区别的,如UART的负电平逻辑,还有UART通信不需要时钟,只需要特定的波特率即可,SPI与IIC都可以有一个主机,多个从机的情况,…

大家好,又见面了,我是你们的朋友全栈君。

 iic通信协议是什么

  IIC协议是二线制,信号线包含SDA和SCL,且信号线是双向的,开路结构,需要通过上拉电阻到VCC,具体的电阻值影响的是信号反应速度和驱动能力。

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

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

  iic通信协议是什么[通俗易懂]

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

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

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

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

  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信号,表示通信结束。

  再来看读的时序:

  iic通信协议是什么[通俗易懂]

  由上图可看出读时序的前面处理方式与写相同,不同的时在第三个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

  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

  仿真结果如下:

  iic通信协议是什么[通俗易懂]

转载于:https://www.cnblogs.com/fire909090/p/11573907.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/133360.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • java 创建新文件_Java创建新文件[通俗易懂]

    java 创建新文件_Java创建新文件[通俗易懂]创建文件是一种非常常见的IO操作,在这一小节中我们将学习如何在java中创建文件的几个方法。在java中创建文件有三种流行的方法,下面将一个一个地来学习。方法一:使用File.createNewFile()方法java.io.File类可用于在Java中创建新文件。当初始化File对象时,需要提供一个文件名,然后调用createNewFile()方法来在Java中创建新文件。如果创建新文件成功,则…

  • Lunix历史及如何学习

    Lunix历史及如何学习1.Lunix是什么1.1Lunix是操作系统还是应用程序Lunix是一套操作系统,它提供了一个完整的操作系统当中最底层的硬件控制与资源管理的完整架构,这个架构是沿袭Unix良好的传统来的,所以相当的稳定而功能强大!Lunix具有核心和系统呼叫两层。Torvalds先生在1991年写出Linux核心的时候,其实该核心仅能『驱动386所有的硬件』而已,所…

  • hashmap面试题简书_三年php面试题

    hashmap面试题简书_三年php面试题这篇文章仅限小编个人的理解,小编不是Java方向的,只是对Java有很高的学习兴趣如果有什么不对的地方还望大佬指点HashMap的底层是数组+链表,(很多人应该都知道了)JDK1.7的是数组+链表(1.7只是一个例子,以前的话也是这样后面就以1.7为例子了)首先是一个数组,然后数组的类型是链表元素是头插法JDK1.8的是数组+链表或者数组+红黑树首先是一个数组,然后数组的类型是链表在链表的元素大于8的时候,会变成红黑树在红黑树的元素小于6的时候会变成链表元素进行尾插HaspM.

  • 两个服务通过http传输excel文件

    两个服务通过http传输excel文件两个服务通过http传输excel文件

  • 大整数的加法运算_大整数相减

    大整数的加法运算_大整数相减问题:如果给出两个很大很大的整数,这两个数大到long类型也装不下,如何求他们的和呢?回顾起小学数学,当我们需要计算两个较大数目的加减乘除,我们是用列竖式的方式来计算的。因为对于较大的整数,我们无法一步就直接计算出结果,所以不得不把计算过程拆分成一个一个小步骤来完成。不仅仅是人脑,对于计算机来说也可以这样解决。程序不可能通过一条指令计算出两个大整数之和,却可以像列竖式一样将运算拆解成若干小…

  • pycharm2020 激活码【中文破解版】

    (pycharm2020 激活码)JetBrains旗下有多款编译器工具(如:IntelliJ、WebStorm、PyCharm等)在各编程领域几乎都占据了垄断地位。建立在开源IntelliJ平台之上,过去15年以来,JetBrains一直在不断发展和完善这个平台。这个平台可以针对您的开发工作流进行微调并且能够提供…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号