PGA之旅设计:IIC通信完整指南
21ic 2024-07-08



一. 简介

这是FPGA之旅设计的第五例啦!今天给大家带来的是IIC通信,IIC协议应用非常广泛,例如与MPU6050进行通信,配置OV5640摄像头、驱动OLED屏幕等等,都需要使用到IIC协议,所以掌握它是非常必要的,废话不多说,接着往下看。文末获取完整代码。



二. IIC简介

IIC协议分为主机和从机,所有的请求都是由主机发出,从机进行响应,从机是没有办法对主机进行读或写的。IIC协议共有两根线,数据线SDA和时钟线SCL,两根线就可以完成所有的通信请求,简直是太给力了。




三. IIC协议

终于到了IIC协议的部分。IIC协议简单来说,共有五种状态,这五种状态的有序组合就组成了完整的IIC通信,学习IIC协议,就是学习这五种状态。

  • 空闲态:  SCL 和 SDA 都为高电平,不进行通信的时候。

  • 起始态:在SCL为高电平的时候,将SDA拉低,主机通知从机,开始进行通信。

  • 数据传输态:数据传输态,又可以分为读和写两个部分,过程都是一样的,就合在一起了,都是在SCL为低电平的时候,SDA将数据发送,在SCL为高电平的时候,将数据接收。

  • (非)应答态:数据传输态完成后,必须接一个应答态或者非应答态,为了确定对方接收到了数据。在SCL为高电平的时候,检测到SDA为低电平,则为应答,否则为非应答。

  • 停止态:一次数据传输完成,由主机发起,在SCL为高电平的时候,SDA由低电平变成高电平。

了解了这五种状态后,接下来就要学习如何使用这五种状态来进行读写操作了。


(一)  IIC写操作

下面就是一个完整的写操作,共包含三次数据传输态,第一次发送的是从机地址 + 0,第二次发送的是寄存器的地址,第三次写的是数据,写入寄存器中的数据。从机地址一般为7bit,与另外一bit共同组成8bit,0表示写,1表示读。

(二)IIC读操作

读操作要比写操作复杂一点,需要的状态多一些。一共有五个数据传输态,状态图如下了。

上面的流程图都是对从机的地址为7位以及从机的寄存器地址为8位的操作。




四. Verilog代码实现

有了上面的各个状态中,SDA和SCL的变换关系,以及读写的序列,就可以很方便的来写程序啦。

1. 首先,当然离不开状态机,根据上面叙述的五种状态,编写状态机,状态机中,将数据传输态分成了读和写两种状态。有了各个状态,操作SDA和SCL两根线不是易如反掌嘛

/*IIC 状态*/localparam IIC_IDLE       = 6'b000_001; /*空闲态*/localparam IIC_START      = 6'b000_010; /*起始态*/localparam IIC_WRDATA     = 6'b000_100; /*写数据态*/localparam IIC_RDDATA     = 6'b001_000; /*读数据态*/localparam IIC_ACK        = 6'b010_000; /*应答态*/localparam IIC_STOP       = 6'b100_000; /*停止态*/


2. 状态机的跳转条件如下,跳转条件和上面叙述的一样。单独看这个有点难懂,有些变量不明白其具体含义,可以结和仿真图形和完整代码进行理解。

/*状态机*/always @(*)begin case(state) IIC_IDLE:  if(IICWriteReq == 1'b1 || IICReadReq == 1'b1) next_state <= IIC_START; else next_state <= IIC_IDLE; IIC_START: if(IICCnt == (IIC_Pre * 'd2)) next_state <= IIC_WRDATA; else next_state <= IIC_START; IIC_WRDATA: if(IICBitCnt == 'd8 && IICCnt == IIC_Pre /4 && iicCLK == 1'b0) next_state <= IIC_ACK; else next_state <= IIC_WRDATA; IIC_RDDATA: if(IICBitCnt == 'd8 && IICCnt == IIC_Pre /4 && iicCLK == 1'b0) next_state <= IIC_ACK; else next_state <= IIC_RDDATA; IIC_ACK: if(IICACKStopCnt == 'd1 && IICCnt == IIC_Pre /4 && iicCLK == 1'b0) if(IICSendBytes == 'd3)  if(IICWriteReq == 1'b1)         /*三个字节发送完成,进入停止态*/ next_state <= IIC_STOP; else  next_state <= IIC_RDDATA; else if(IICSendBytes == 'd2 && IICReadReq == 1'b1) next_state <= IIC_START; else if(IICSendBytes == 'd4) next_state <= IIC_STOP; else next_state <= IIC_WRDATA; else next_state <= IIC_ACK; IIC_STOP: if(IICACKStopCnt == 'd1 && IICCnt == IIC_Pre/4 && iicCLK == 1'b1) next_state <= IIC_IDLE; else next_state <= IIC_STOP; default: next_state <= IIC_IDLE; endcaseend

各个部分实现的详细代码,就不列举出来啦,代码总计280多行,也不算多。通过本IIC模块,可以驱动OV5640摄像头,MPU6050模块和0.96寸OLED屏幕等等,后续会基于此模块,来驱动这些外设。




五. testbeach编写

还是按照流程走,编写完模块后,进行一下仿真,还真有错误,幸亏仿真了,哈哈哈。

`timescale 1ns/1ps module testbench();  reg clk; reg rst; wire SDA; wire SCL; reg IICWriteReq; reg IICReadReq; wire IICWriteDone; wire IICReadDone; always # 50 clk = ~clk; initial begin clk = 1'b1; rst = 1'b1;  IICWriteReq = 1'b0; IICReadReq = 1'b1; #100   /*手动复位*/ rst = 1'b0; #100 rst = 1'b1; end  always@(posedge clk) if(IICReadDone == 1'b1)   /*读完成后,readReq为0,只进行一次读写操作*/ IICReadReq <= 1'b0; else IICReadReq <= IICReadReq; IIC_Driver IIC_DriverHP( .sys_clk (clk),           /*系统时钟*/ .rst_n (rst),             /*系统复位*/  .IICSCL (SCL),            /*IIC 时钟输出*/ .IICSDA (SDA),             /*IIC 数据线*/  .IICSlave ('h1234),  .IICWriteReq (IICWriteReq),       /*IIC写寄存器请求*/ .IICWriteDone (IICWriteDone),      /*IIC写寄存器完成*/ .IICWriteData ('h5a), /*IIC发送数据 8bit的从机地址 + 8bit的寄存器地址 + 8bit的数据(读忽略,后默认为0)*/  .IICReadReq (IICReadReq),        /*IIC读寄存器请求*/ .IICReadDone (IICReadDone),       /*IIC读寄存器完成*/ .IICReadData ()/*IIC读取数据*/); endmodule
声明: 本文转载自其它媒体或授权刊载,目的在于信息传递,并不代表本站赞同其观点和对其真实性负责,如有新闻稿件和图片作品的内容、版权以及其它问题的,请联系我们及时删除。(联系我们,邮箱:evan.li@aspencore.com )
0
评论
  • 相关技术文库
  • EDA
  • 仿真
  • CAD
  • 芯片
  • 芯片散热怎么设计?

    在IC中,一个温度参照点始终是器件的一个节点,即工作于给定封装中的芯片内部最热的点。

    4小时前
  • 如何选择全定制与半定制ASIC设计

    ASIC分为全定制和半定制。全定制设计需要设计者完成所有电路的设计,因此需要大量人力物力,灵活性好但开发效率低下。如果设计较为理想,全定制能够比半定制的ASIC芯片运行速度更快。半定制使用库里的标准逻辑单元(...

    09-12
  • IGMPv2的组播路由器选举机制的优势是什么?

    IGMPv2 igmp版本2对版本1所做的改进主要有:(1)共享网段上组播路由器的选举机制共享网段表示一个网段上有多个组播路由器的情况。在这种情况下,由于此网段上运行igmp的路由器都能从主机那里收到成员资格报告消息,...

    09-12
  • 物理地址在不同计算机体系中的表现如何?

    在存储器里以字节为单位存储信息,为正确地存放或取得信息,每一个字节单元给以一个唯一的存储器地址,称为物理地址(Physical Address),又叫实际地址或绝对地址。 地址从0开始编号,顺序地每次加1,因此存储器的...

    09-12
  • 假芯片对电子行业的影响分析

    芯片的重要性不言而喻,可以说,一切智能电子设备都离不开芯片。但是,由于目前芯片相对短缺,因此市场上出现了部分假芯片。为增进大家对芯片的认识,本文将对假芯片的相关内容予以介绍。如果你对芯片具有兴趣,不...

    09-12
  • 语音芯片在儿童玩具中的创新使用

    芯片是一种非常重要的电子器件,可以说,芯片制作能力强的企业或者国家都占据着重要地位。为增进大家对芯片的认识,本文将对语音芯片的应用范围、语音芯片的选择予以介绍。如果你对语音芯片具有兴趣,不妨继续往下...

    09-12
  • 语音芯片在日常生活中的应用有哪些?

    芯片在诸多产品中都有所应用,如手机处理器芯片、蓝牙芯片、AI芯片等等。虽然我们可能对芯片的底层知识并不了解,但是对于芯片的作用,还总是能说出个一二三。为增进大家对芯片的认识,本文将对语音芯片如何录音、...

    09-04
  • 压力变送器的常见故障及解决方案是什么?

    压力变送器在工业环境中具有重要应用,对于压力变送器,电气等专业的朋友相对更为熟悉。为增进大家对压力变送器的认识,本文将基于两点介绍压力变送器:1、压力变送器的日常维护,2、压力变送器如何校准。如果你对...

    09-04
  • 电子元器件散热方法的常见误区

    元器件是电子领域的基本元件,很多重要的设备都是通过元器件构建起来的。为增进大家对电子元器件的认识,本文将对元器件的几种常见的散热方法予以详细介绍。如果你对元器件具有兴趣,不妨和小编一起来继续往下阅读...

    08-26
  • 熔断器与断路器的选择性比较

    熔断器和断路器在主要性能方面存在一些差异,为了更好选择和使用熔断器和断路器,下面针对熔断器和断路器的主要性能进行对比。

    08-13
  • MATLAB仿真RLC电路基础教程

    本文演示如何用MATLAB进行电路仿真,测量RLC电路的电压。我用的是R2014a,不同版本软件界面

    08-09
下载排行榜
更多
评测报告
更多
广告