1. 起始结束信号的判断

  1. //---------------------------------------------
  2. //start,stop condition judgement
  3. //---------------------------------------------
  4. wire start, stop;
  5. reg sda1, sda2;
  6. reg sda11;
  7. always @ ( posedge SCL )   //触发器1
  8. sda1 <= SDA;
  9. always @ ( negedge SCL )   //触发器2
  10. sda2 <= SDA;
  11. always @ ( negedge SCL )   //触发器3
  12. sda11 <= sda1;
  13. assign start = sda11 & (!sda2);
  14. assign stop = sda2 & ( !sda11 );

判断的想法:当SDA上传输正常的数据信号时,只有在SCL低电平时SDA发生电平变化,所以正常的数据信号电平保持时间是SCL周期的整数倍,即1bit数据信号的SDA电平用SCL的上升沿和下降沿采集得到的是相同的电平;而起始和结束信号是在SCL的高电平时刻发生变化,即在一个SCL周期内,SCL的上升沿和下降沿将采集到不同的SDA电平。基于此特点,采用下面的触发器方法,来判断起始和结束信号。

分别用SCL的上升沿和下降沿做触发器出发信号,去采集SDA信号,如果是SDA上是数据信号,触发器的输出将是SDA的延迟。用触发器1来说,如果SDA上是数据信号则得到的sda1信号波形与SDA波形一样,只是时间上比SDA延迟四分之一个SCL周期。相同用SCL下降沿采集的触发器2输出sda2的波形,是延迟四分之三个SCL周期的SDA波形。再用触发器3将sda1延迟二分之一个SCL周期后,得到的输出sda11波形将和sda2的波形完全重合。

这时用sda11与sda2比较,如果SDA传的是数据信号,则sda11与sda2的波形将完全一样,但在SDA有起始或结束信号的位置,sda11的波形和sda2的波形会不一样,根据不同的与非组合,就可以得到start和stop信号。

2. 设置计数器

  1. //----------------------------------------------
  2. //count setting
  3. //----------------------------------------------
  4. reg [3:0]  bitcont;
  5. wire bit_ack = bitcont[3];
  6. always @ ( posedge SCL or posedge start)
  7. begin
  8. if ( start )
  9. bitcont <=  4'h6;
  10. else
  11. begin
  12. if (bit_ack)
  13. bitcont <= 4'h6;
  14. else
  15. bitcont <= bitcont -4'h1;
  16. end
  17. end


3. Match地址

  1. //----------------------------------------
  2. //address match
  3. //----------------------------------------
  4. reg addr_match, op_read;
  5. always @ ( negedge SCL or posedge start )
  6. begin
  7. if ( start )
  8. begin
  9. addr_match <= 1'h1;
  10. op_read <= 1'h0;
  11. end
  12. else
  13. begin
  14. if( (bitcont == 6) & (sdar != I2C_ADR[6])) addr_match <= 1'h0;
  15. if( (bitcont == 5) & (sdar != I2C_ADR[5])) addr_match <= 1'h0;
  16. if( (bitcont == 4) & (sdar != I2C_ADR[4])) addr_match <= 1'h0;
  17. if( (bitcont == 3) & (sdar != I2C_ADR[3])) addr_match <= 1'h0;
  18. if( (bitcont == 2) & (sdar != I2C_ADR[2])) addr_match <= 1'h0;
  19. if( (bitcont == 1) & (sdar != I2C_ADR[1])) addr_match <= 1'h0;
  20. if( (bitcont == 0) & (sdar != I2C_ADR[0])) addr_match <= 1'h0;
  21. if( bitcont == 0 ) op_read <= sdar;
  22. end
  23. end

4. 发送ACK

  1. //send ack
  2. //-----------------------------------------------------------------------
  3. reg ack_assert;
  4. always @ ( negedge SCL )
  5. begin
  6. if ( bit_ack & addr_match & op_read )
  7. ack_assert <= 1'h1;
  8. else
  9. ack_assert <= 1'h0;
  10. end
  11. //-------------------------------------------------------------------------
  12. //control SDA line
  13. //-------------------------------------------------------------------------
  14. assign SDA = ack_assert ? 1'h0 : 1'hz;
  15. pullup ( SDA );

总结:

在实际应用中,此程序有局限性,这里假设I2C起始信号和结束信号的变化超过一个SCL高电平时间,如果起始信号或结束信号,以一个拉低脉冲或拉高脉冲的的形式出现,用SCL的上升沿和下降沿是采不到SDA的变化的。

I2C Verilog的实现(二)的更多相关文章

  1. 【第一季】CH05_FPGA设计Verilog基础(二)Enter a post title

    [第一季]CH05_FPGA设计Verilog基础(二) 5.1状态机设计 状态机是许多数字系统的核心部件,是一类重要的时序逻辑电路.通常包括三个部分:一是下一个状态的逻辑电路,二是存储状态机当前状态 ...

  2. CRC校验原理和verilog实现方法(二)

    1 前言 在 前面的博客  CRC校验原理和verilog实现方法(一)  中,介绍了CRC校验的原理和手动计算过程.本文说一下我在学习CRC校验FPGA实现的一点心得体会. 2 线性反馈移位寄存器 ...

  3. I2C控制器的Verilog建模之二

    前言:接着上一篇的I2C写操作,今天要实现一个I2C的读操作.虽然在ADV7181B配置内部寄存器时没有必要使用到读操作,但是为了进一步确认寄存器是否在I2C写模块下被正确配置,这一步是必不可少的. ...

  4. System Verilog基础(二)

    这一篇笔记主要记录Procedural,Process,Task and function,Interface和Communication中值得注意的点. 1.Procedural 写testbenc ...

  5. Norflash控制器的Verilog建模之二(仿真)

    前言:经过几天修改,norflash控制器基本已经完成,通过仿真.完整的norflash包含2个模块:直接操作硬件的norflash_ctrl.v与控制ctrl模块的驱动norflash_driver ...

  6. I2C Verilog的实现(一)

    <span style="font-size:14px;">`timescale 1ns / 1ps module test( sda ); reg scl; inou ...

  7. System Verilog MCDF(二)

    整形器的接口时序: reg,grant是维持了两个clk的. chid ,length在发送数据期间不可以变化. 第一个data数据必须在start上升沿的同一个clk发送. reg,grant两者之 ...

  8. Smart210学习记录-----Linux i2c驱动

    一:Linux i2c子系统简介: 1.Linux 的 I2C 体系结构分为 3 个组成部分: (1) I2C 核心. I2C 核心提供了 I2C 总线驱动和设备驱动的注册.注销方法,I2C 通信方法 ...

  9. 分享:FIFO 同步、异步以及Verilog代码实现

    FIFO 很重要,之前参加的各类电子公司的逻辑设计的笔试几乎都会考到. FIFO是英文First In First Out 的缩写,是一种先进先出的数据缓存器,他与普通存储器的区别是没有外部读写地址线 ...

随机推荐

  1. 【转】IOS 怎么获取外设的广播数据AdvData

    原文网址:http://www.deyisupport.com/question_answer/wireless_connectivity/bluetooth/f/103/t/73443.aspx N ...

  2. 在Sharepoint2010中发布VS2010开发的WebPart

    转:http://www.cnblogs.com/bfgl/archive/2012/03/22/2411698.html 本人接触Sharepoint2010到今天为止不到一个月.作为一名老C#程序 ...

  3. cannot find the word template:WordToRqm.dot的解决方法

    powerdesigner安装后: C:"Program Files"Sybase"PowerDesigner 12"Add-ins"Microsof ...

  4. HDU 1518

    思路:从第一个数开始搜索,将其和与边长比对,相等则计数+1,计数达到3的时候说明可以组成,因为剩下那条必与边长相等,搜索过程注意剪枝,若某个数已被加入边长则不能重复计算,应将其标记,另外应在每一层递归 ...

  5. 使用vxsim(一)

    新建download工程 选择toolchain 这里选SIMNTgnu是为了在vxsim中能下载编译的.o文件 如果是为了下载到visualbox或者vmware中的vxworks,则toolcha ...

  6. Codeforces Round #343 (Div. 2) D - Babaei and Birthday Cake 线段树+DP

    题意:做蛋糕,给出N个半径,和高的圆柱,要求后面的体积比前面大的可以堆在前一个的上面,求最大的体积和. 思路:首先离散化蛋糕体积,以蛋糕数量建树建树,每个节点维护最大值,也就是假如节点i放在最上层情况 ...

  7. 可在Azure上运行.NET的SDK

    本节包含前几个版本的 for .NET 的发行说明.有关早期版本的详细信息,请参阅早期版本:Azure SDK for .NET.有关 SDK 支持和停用的信息,请参阅 Azure SDK for . ...

  8. 非阻塞,send后马上close消息能成功发出去吗

    send返回成功的话就能. 判断send返回,假如返回的值是成功,则说明要发送的放到了tcp的发送缓冲区内了,然后在close的话等于再放了一个关闭信号到缓冲区里的. 由于TCP是有序的,假如对端此时 ...

  9. java枚举enum

    http://www.cnblogs.com/wenruo/p/5349614.html java的枚举通过关键字enum实现.可以理解为一个类,不过这个类由编译器自动加了一些方法. static v ...

  10. PC-ADSL开机自动拨号方法

    方法一:把adsl拨号的快捷方式放到“开始”菜单中“所有程序”中的“启动”中. 再到“控制面板”中的“网络连接”中找到你用的拨号连接,鼠标右键点击选择“属性”.然后,在窗口上部选择“选项”,把“拨号选 ...