基于FPGA的以太网设计(三)

news2025/1/18 8:45:38

通过前文介绍了RGMII接口时序我们可以知道,RGMII接口是在时钟信号的上升沿和下降沿均进行数据的传输,而FPGA则在时钟的单沿传输数据,因此我们需要编写代码将RGMII接口转换为GMII接口。

由于前面的介绍我们知道RTL8211默认工作在延时状态,并且可以通过控制上拉或者下拉来改变延时与否,如下图所示。

发送的时序图如下所示:

可以看到当PHY芯片工作在延时模式的时候,会将发送端的时钟进行2ns的延时作为接收端的数据接收时钟与控制时钟,这也是最常用的工作模式。RTL8211FD默认就是工作在延时状态,因此不再需要MDIO接口对寄存器进行配置。

实现RGMII与GMII需要用到两个原语分别为ODDR和IDDR,接下来我们先简单了解一下两个原语的用法。

  1. LOGIC

ODDR位于OLOGIC中,可以把单沿传输的数据转换为双沿传输的数据 在讲解ODDR功能之前,需要先了解OLOGIC的结构及功能。OLOGIC块位于IOB的内侧,FPGA内部信号想要输出到管脚,都必须经过OLOGIC。OLOGIC资源的类型为OLOGICE2(HP I/O Bank)和OLOGICE3(HR I/O Bank),两者在功能和结构上是相同的,所以本文统称为OLOGIC,其结构如下图所示。

如果输出的信号不使用OLOGIC中的ODDR功能,那么此时信号从上图中红线路径进行传输,从组合逻辑电路输出到IOB模块。如果要使用OLOGIC模块中的D触发器功能,那么信号从D1进入OLOGIC模块,沿绿色信号线进行传输。如果要使用OLOGIC的ODDR功能,把单沿传输的信号转换为双沿传输的信号,此时需要两个输入信号D1、D2沿蓝色路径进行传输。

ILOGICE位于IOB旁边,ILOGICE块包含同步元件,用于在数据通过IOB进入FPGA时捕获数据。7系列芯片中ILOGICE可能是ILOGICE2(HP I/O bank)或ILOGICE3(HR I/O bank)。

当需要使用IDDR这些转换功能时,通过绿色的信号传输到下面器件的输入端D,这个器件可以配置为锁存器(latch)、触发器(FF)、双沿转单沿器件(DDR),而CE1是时钟使能信号,高电平有效,不连接时默认为高电平。

当不使用IDDR功能时,在使用IDDR原语后,只需要添加(IOB == “TRUE”)原语,就可以使用ILOGICE中的触发器(FF)的功能。 这个触发器相比FPGA内部触发器更靠近FPGA管脚,使得建立时间余量更大,更有利于时序。

Q1和Q2是输出端口,S/R是复位、置位端口,在调用原语是可以对同步、异步触发进行设置。此框图中S/R的是一根线,但IDDR的原语将S和R分开为两个端口,使用时不能让S、R同时有效。

1.ODDR

ODDR的框图如下所示:

端口名

含义

C

时钟输入信号。

CE

时钟使能信号,高电平有效。

D1D2

ODDR输入信号。

S/R

置位/复位引脚,高电平有效。

Q

ODDR输出信号。

ODDR原语的参数如下图所示:

参数名

含义

取值

DDR_CLK_EDGE

ODDR工作模式

OPPOSITE_EDGE (默认), SAME_EDGE

INIT

设置Q端口的初始值

0(默认),1

SRTYPE

设置复位/置位相对于时钟的类型

ASYNC, SYNC(默认)

ODDR两种工作模式的时序图前面都已经提到过这里就不再赘述。

ODDR原语例化模版如下:

ODDR #(

        .DDR_CLK_EDGE     ("OPPOSITE_EDGE"), // "OPPOSITE_EDGE" or "SAME_EDGE"

        .INIT             (1'b0),    // Initial value of Q: 1'b0 or 1'b1

        .SRTYPE           ("SYNC") // Set/Reset type: "SYNC" or "ASYNC"

   ) ODDR_inst (

        .Q                (Q),   // 1-bit DDR output

        .C                (C),   // 1-bit clock input

        .CE               (CE), // 1-bit clock enable input

        .D1               (D1), // 1-bit data input (positive edge)

        .D2               (D2), // 1-bit data input (negative edge)

        .R                (R  ),   // 1-bit reset

        .S                (S  )    // 1-bit set

   );

其实该原语的使用与理解都不难,只需要记住它的功能是将TXC同一个时钟周期内的两个SDR信号分别通过上升沿和下降沿输出为DDR信号。

2.IDDR

IDDR其实可以看做ODDR的逆过程,FPGA在接收DDR信号时可以利用IDDR原语将其转化为SDR信号,以便FPGA内部进行处理。

IDDR框图如下所示:

端口名

含义

Q1Q2

IDDR输出寄存器

C

时钟输入引脚

CE

必须为高电平才能将新数据加载到DDR触发器中,低电平时,时钟转换被忽略,新数据不会加载到DDR触发器中。

D

IDDR寄存器从IOB输入

S/R

同步/异步设置/复位引脚,高电平有效。

模版如下:

   IDDR #(

      .DDR_CLK_EDGE("OPPOSITE_EDGE"), // "OPPOSITE_EDGE", "SAME_EDGE"

                                      //    or "SAME_EDGE_PIPELINED"

      .INIT_Q1(1'b0), // Initial value of Q1: 1'b0 or 1'b1

      .INIT_Q2(1'b0), // Initial value of Q2: 1'b0 or 1'b1

      .SRTYPE("SYNC") // Set/Reset type: "SYNC" or "ASYNC"

   ) IDDR_inst (

      .Q1(Q1), // 1-bit output for positive edge of clock

      .Q2(Q2), // 1-bit output for negative edge of clock

      .C(C),   // 1-bit clock input

      .CE(CE), // 1-bit clock enable input

      .D(D),   // 1-bit DDR data input

      .R(R),   // 1-bit reset

      .S(S)    // 1-bit set

   );

3.GMII转RGMII

具体实现代码如下图所示

`timescale 1ns / 1ps

module gmii2rgmii(
    input                           gmii_txclk      ,       //GMII发送时钟
    input                           gmii_tx_en      ,       //GMII发送使能
    input                           gmii_tx_er      ,       //GMII发送错误
    input           [7:0]           gmii_tx_data    ,       //GMII发送数据

    output                          rgmii_txclk     ,       //RGMII发送时钟
    output                          rgmii_tx_ctl    ,       //RGMII发送控制
    output          [3:0]           rgmii_tx_data           //RGMII发送数据
);

    //时钟转换
    ODDR #(
        .DDR_CLK_EDGE     ("SAME_EDGE"      ),      // "OPPOSITE_EDGE" or "SAME_EDGE" 
        .INIT             (1'b0                 ),      // Initial value of Q: 1'b0 or 1'b1
        .SRTYPE           ("SYNC"               )       // Set/Reset type: "SYNC" or "ASYNC" 
   ) ODDR_inst_2 (
        .Q                (rgmii_txclk          ),      // 1-bit DDR output
        .C                (gmii_txclk           ),      // 1-bit clock input
        .CE               (1'b1                 ),      // 1-bit clock enable input
        .D1               (1'b1                 ),      // 1-bit data input (positive edge)
        .D2               (1'b0                 ),      // 1-bit data input (negative edge)
        .R                (1'b0                 ),      // 1-bit reset
        .S                (1'b0                 )       // 1-bit set
   );

    //数据转换
    genvar i;
    generate
        for (i=0; i < 4; i=i+1)
        begin: rgmii_txd_o
        ODDR #(
            .DDR_CLK_EDGE     ("SAME_EDGE"  ),      // "OPPOSITE_EDGE" or "SAME_EDGE" 
            .INIT             (1'b0             ),      // Initial value of Q: 1'b0 or 1'b1
            .SRTYPE           ("SYNC"           )       // Set/Reset type: "SYNC" or "ASYNC" 
        ) ODDR_inst_0 (
            .Q                (rgmii_tx_data[i] ),      // 1-bit DDR output
            .C                (gmii_txclk       ),      // 1-bit clock input
            .CE               (1'b1             ),      // 1-bit clock enable input
            .D1               (gmii_tx_data[i]  ),      // 1-bit data input (positive edge)
            .D2               (gmii_tx_data[i+4]),      // 1-bit data input (negative edge)
            .R                (1'b0             ),      // 1-bit reset
            .S                (1'b0             )       // 1-bit set
        );
      end
    endgenerate

    //控制信号转换
    ODDR #(
        .DDR_CLK_EDGE     ("SAME_EDGE"      ),      // "OPPOSITE_EDGE" or "SAME_EDGE" 
        .INIT             (1'b0                 ),      // Initial value of Q: 1'b0 or 1'b1
        .SRTYPE           ("SYNC"               )       // Set/Reset type: "SYNC" or "ASYNC" 
    ) ODDR_inst_1 (
        .Q                (rgmii_tx_ctl         ),      // 1-bit DDR output
        .C                (gmii_txclk           ),      // 1-bit clock input
        .CE               (1'b1                 ),      // 1-bit clock enable input
        .D1               (gmii_tx_en           ),      // 1-bit data input (positive edge)
        .D2               (gmii_tx_er^gmii_tx_en),      // 1-bit data input (negative edge)
        .R                (1'b0                 ),      // 1-bit reset
        .S                (1'b0                 )       // 1-bit set
    );

endmodule
`timescale 1ns / 1ps

module gmii2rgmii_tb();
    reg            gmii_txclk      ;
    reg            gmii_tx_en      ;
    reg            gmii_tx_er      ;
    reg     [7:0]  gmii_tx_data    ;
    
    wire           rgmii_txclk     ;
    wire           rgmii_tx_ctl    ;
    wire    [3:0]  rgmii_tx_data   ;

    initial gmii_txclk = 1'b1;
    always#5 gmii_txclk = ~gmii_txclk;

    initial begin
        gmii_tx_en  = 0;
        gmii_tx_er  = 0;
        gmii_tx_data = 8'd0;
        #201;
        repeat(100)begin
            @(posedge gmii_txclk)
                gmii_tx_data = {$random} % 256;  
                gmii_tx_en = 1'b1;
                gmii_tx_er = 1'b0; 
        end
        #100;
        $stop;
    end
    
    gmii2rgmii u_gmii2rgmii(
        .gmii_txclk                 (gmii_txclk   ),
        .gmii_tx_en                 (gmii_tx_en   ),
        .gmii_tx_er                 (gmii_tx_er   ),
        .gmii_tx_data               (gmii_tx_data ),
        .rgmii_txclk                (rgmii_txclk  ),
        .rgmii_tx_ctl               (rgmii_tx_ctl ),
        .rgmii_tx_data              (rgmii_tx_data)
    );

endmodule

仿真结果如下:

  1. 4.RGMII转GMII

具体代码如下所示:

`timescale 1ns / 1ps

module rgmii2gmii(
    input                       rgmii_rxclk     ,   //RGMII接收时钟
    input                       rgmii_rx_ctl    ,   //RGMII接收控制
    input           [3:0]       rgmii_rx_data   ,   //RGMII接收数据

    output                      gmii_rxclk      ,   //GMII接收时钟
    output                      gmii_rx_dv      ,   //GMII接收有效
    output          [7:0]       gmii_rx_data        //GMII接收数据
);

    wire            [4:0]       din             ;   //控制信号与数据进行拼接
    wire            [9 : 0]     gmii_data       ;   //转换完成后的信号

    assign din = {rgmii_rx_ctl,rgmii_rx_data}   ;
    assign gmii_rxclk = rgmii_rxclk             ;

    genvar i;
    generate
      for (i=0; i < 5; i=i+1)
      begin: rgmii_rxd_i
    IDDR #(
      .DDR_CLK_EDGE             ("SAME_EDGE_PIPELINED"  ), // "OPPOSITE_EDGE", "SAME_EDGE" 
                                                           //    or "SAME_EDGE_PIPELINED" 
      .INIT_Q1                  (1'b0                   ), // Initial value of Q1: 1'b0 or 1'b1
      .INIT_Q2                  (1'b0                   ), // Initial value of Q2: 1'b0 or 1'b1
      .SRTYPE("SYNC")                                      // Set/Reset type: "SYNC" or "ASYNC" 
    ) IDDR_inst (
      .Q1                       (gmii_data[i]        ), // 1-bit output for positive edge of clock
      .Q2                       (gmii_data[i+5]      ), // 1-bit output for negative edge of clock
      .C                        (rgmii_rxclk            ), // 1-bit clock input
      .CE                       (1'b1                   ), // 1-bit clock enable input
      .D                        (din[i]                 ), // 1-bit DDR data input
      .R                        (1'b0                   ), // 1-bit reset
      .S                        (1'b0                   )  // 1-bit set
    );
      end  
    endgenerate

    assign gmii_rx_data = {gmii_data[8:5],gmii_data[3:0]};
    assign gmii_rx_dv = gmii_data[4] & gmii_data[9];//

endmodule
`timescale 1ns / 1ps

module rgmii2gmii_tb();
    reg                 rgmii_rxclk  ;
    reg                 rgmii_rx_ctl ;
    reg        [3:0]    rgmii_rx_data;
    reg                 rgmii_rxclk_r;

    wire                gmii_rxclk   ;
    wire                gmii_rx_dv   ;
    wire       [7:0]    gmii_rx_data ;
    

    initial begin#2; rgmii_rxclk = 1'b1;
    forever #4 rgmii_rxclk = ~rgmii_rxclk;
    end

    initial  rgmii_rxclk_r = 1'b1;
    always#4 rgmii_rxclk_r = ~rgmii_rxclk_r;

    initial begin
        rgmii_rx_ctl = 1'b0;
        rgmii_rx_data = 4'd0;
        #202;
        repeat(100)begin
            @(posedge rgmii_rxclk_r) 
            rgmii_rx_ctl = 1'b1;
            rgmii_rx_data = {$random} % 16;
            #4;
            rgmii_rx_data = {$random} % 16;
        end
        #100;
        $stop;
    end

    rgmii2gmii u_rgmii2rgmii(
        .rgmii_rxclk                (rgmii_rxclk  ),
        .rgmii_rx_ctl               (rgmii_rx_ctl ),
        .rgmii_rx_data              (rgmii_rx_data),
        .gmii_rxclk                 (gmii_rxclk   ),
        .gmii_rx_dv                 (gmii_rx_dv   ),
        .gmii_rx_data               (gmii_rx_data )
    );

endmodule

仿真结果如下:

需要注意的是仿真是需要让时钟在上升沿和下降沿对齐数据的中心,不然后出现如下图所示的情况。

参考:xilinx原语详解及仿真之ODDR_oddr原语使用说明-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2214322.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

体能攻略梳理

最近在锻炼身体&#xff0c;但如何才能更好的锻炼&#xff1f;最近看了慕课上的体能攻略&#xff08;https://www.icourse163.org/learn/HZIC-1206696841&#xff09;&#xff0c;讲的特别好。希望大家看了就是会了&#xff0c;看了就是做了&#x1f601;。 基本知识 减脂别只…

数据库常见面试

8道面试题 目录 目录 7道面试题 1.怎样进行sql优化 4、group by优化 5、limit优化 6、count优化 7、update优化 2.。怎样查看sql执行情况呢(哪个关键字)&#xff0c;说说你对这个关键字的认识 4) possible_key&#xff1a; 5) key 3.说说你对innodb和 myisam的理解 …

pdf阅读器哪个好用?5个软件帮助你快速阅读pdf文件

pdf阅读器哪个好用&#xff1f;5个软件帮助你快速阅读pdf文件 如果你在寻找好用的 PDF 阅读器&#xff0c;有很多强大的软件可以帮助你轻松、高效地阅读和处理 PDF 文件。这些软件不仅可以简单地查看文件&#xff0c;还能提供标注、评论、注释和文档管理等额外功能。以下是5款…

请确保已在git上配置你的user.name和user.email

问题&#xff1a;使用vscode在远程服务器上暂存修改报错&#xff1a; 原因&#xff1a;未在远程服务器上配置该项目对应的git的username和useremail 解决方法&#xff1a; 在vscode中新建一个终端 命名&#xff1a; git config --global user.email "youexample.com&qu…

Linux集群化软件安装,前置

1.准备多个虚拟机 可以选择克隆方式,更快捷 2.配置各个主机IP地址 命令 ; vim /etc/sysconfig/network-scripts/ifcfg-ens33 进入这个文件,操作请看我的文章Linux更改固定IP地址https://blog.csdn.net/m0_72898512/article/details/142718753?spm1001.2014.3001.5501 各个主…

个人博客搭建 | Hexo框架

文章目录 1.Hexo安装2.创建博客3.将博客通过GitHub来部署4.更换主题 1.Hexo安装 Hexo 是一个快速、简洁且高效的博客框架。Hexo 使用 Markdown&#xff08;或其他标记语言&#xff09;解析文章&#xff0c;在几秒内&#xff0c;即可利用靓丽的主题生成静态网页。搭建Hexo首先要…

采购管理系统有哪些功能?

对于制造型企业来说&#xff0c;采购成本占据重要位置&#xff0c;直接影响企业利润空间&#xff0c;低采购成本可提升企业竞争力&#xff0c;高采购成本则可能压缩企业生存发展空间。一款好用的采购管理系统可以帮助企业解决降低成本的难题&#xff0c;那么采购管理系统有哪些…

如何轻松上手低代码和零代码平台进行应用开发?

在当今数字化转型加速的时代背景下&#xff0c;企业与个人对于快速开发高效的应用程序的需求日益增长。低代码&#xff08;Low Code&#xff09;和零代码&#xff08;No Code&#xff09;平台应运而生&#xff0c;为用户提供了无需深入了解编程语言&#xff0c;即可创建复杂业务…

javaweb-xml映射文件编写sql语句

可以使用注解的方式&#xff0c;也可以使用xml映射的方式&#xff0c;一般简单sql语句使用注解&#xff0c;复杂的使用xml映射。

上厕所也要看的 一篇博客学会线程 进程 并行 并发 Thread Runnable等

线程相关概念 程序 为完成特点的任务&#xff0c;用某种语言编写的一组指令的集合&#xff0c;简单来说就是我们的代码 进程 定义&#xff1a;进程是指运行中的程序&#xff0c;比如我们使用QQ,就启动了一个进程&#xff0c;操作系统就会为该进程分配内存空间&#xff0c;当我…

Idea中,resources下面建文件夹踩坑

Idea中&#xff0c;resources下面建文件夹踩坑 直接上问题 mybatis无法加载xml&#xff0c;别划走&#xff0c;这个无法加载可不是配置写错了&#xff0c;继续看&#xff01; 首先看上图中的xml&#xff0c;可以确定的是项目中所有的配置都是正确的&#xff0c;但是运行mappe…

Apache Kafka各Api模块说明

Kafka API 微信公众号&#xff1a;阿俊的学习记录空间小红书&#xff1a;ArnoZhangwordpress&#xff1a;arnozhang1994博客园&#xff1a;arnozhangCSDN&#xff1a;ArnoZhang1994 Kafka 包含五个核心 API&#xff1a; Producer API 允许应用程序将数据流发送到 Kafka 集群中…

在java 中如何实现执行linux命令,通过post接口代理出来?

接口方式输入命令得返回结果 public AjaxResult doPost(HttpServletRequest request, HttpServletResponse response, String command) throws ServletException, IOException {// 设置响应内容类型 text/plain // response.setContentType("application/json"…

Linux——Harbor: 容器镜像的存储

K8s 如何通过harbor 拉取镜像&#xff1f; K8S 在那些情况下需要进行镜像的拉取&#xff1f; 在需要进行新的pod的调度时&#xff0c;基于镜像拉取策略&#xff0c;完成镜像的获取&#xff1a; Always: 在任何情况下都需要重新拉取镜像 即使设置为总是进行镜像拉取&#xff…

ERP系统是什么?ERP系统如何与数据库对接?

ERP系统的定义 1.企业ERP系统标准的定义来自于其英文原意&#xff0c;即企业资源规划(Enterprise Resource Planning)。企业资源计划系统是一种集成的软件系统&#xff0c;旨在帮助企业管理其资源。它可以协调各种不同的业务流程&#xff0c;例如供应链管理、采购、库存管理、…

arm架构ceph pacific部署

背景 合作伙伴实验室的华为私有云原来使用单点的nfs做为存储设备&#xff0c;现有两方面考量&#xff0c;业务需要使用oss了&#xff0c;k8s集群及其他机器也需要一套可扩展的分布式文件系统 部署ceph 初始机器配置规划 IP配置主机名Role10.17.3.144c8g1T数据盘ceph-node01…

2-122 文章复现:基于matlab的多智能体系统一致性算法的电力系统分布式经济调度策略

文章复现&#xff1a;基于matlab的多智能体系统一致性算法的电力系统分布式经济调度策略&#xff0c;应用多智能体系统中的一致性算法&#xff0c;以发电机组的增量成本和柔性负荷的增量效益作为一致性变量&#xff0c;设计一种用于电力系统经济调度的算法&#xff0c;通过分布…

透过3大海外案例,看百亿中老年眼镜市场创新发展方向

国内老花镜市场容量为342.15亿元 前言 全球人口老龄化的加速推进&#xff0c;中老年群体正逐渐成为消费市场中不可忽视的重要力量&#xff0c;每一个细微的市场变化都映射着社会结构与消费需求的深刻转型。 其中&#xff0c;中老年眼镜市场的新场景、新需求不断涌现&#xf…

西门子S7-200 SMART选型指南之产品订货信息

提供了S7-200 SMART控制器的产品订货号列表&#xff0c;用户可以根据订货号准确选择所需的产品型号。 S7-200SMART CPU 订货号 产品选型或方案确认需求需要结合工艺、应用场合等因素综合考虑&#xff0c;最终方案或订货号请与西门子销售或经销商确认。本文列出的订货号仅供参考…

Vite + Vue3 使用 cdn 引入依赖,并且把外部 css、js 文件内联引入

安装插件 pnpm i element-plus echarts axios lodash -S在 vite.config.js 引用 注意事项&#xff1a;element-plus 不能在 vite.config.js 中使用按需加载&#xff0c;需要在 main.js 中全局引入&#xff1b; import { resolve } from path import { defineConfig } from v…