【数集项目之 MCDF】(四) 整形器 formatter

news2025/1/22 17:49:33

  根据上一章的arbiter结构图,结合设计文档中MCDF的整体结构图,可以发现formatter整形器模块是arbiter的上级,负责最终的数据输出,与外界数据接收端相连。

第一节 fromatter文档理解

  设计文档formatter的部分时序介绍如下

如图所示。整形器按照数据包的形式发送数据。数据包的长度可以是4、8、16和32。整形器必须完整的发送某一个通道的数据包后,才可以准备发送下一个数据包。在发送数据包期间,fmt_chidfmt_length应保持不变,直到数据包发送结束。
在这里插入图片描述
整形器准备发送数据包时,先将fmt_req置高,等待接收端的fmt_grant信号。当fmt_grant信号变为高时,则在下一个周期将:
(1) fmt_req置低
(2) fmt_start输出一个时钟周期的高电平脉冲
(3) fmt_data开始送出第一个数据
数据开始发送后应连接发送,中间不允许有空闲周期,直到发送完所有数据。在发送最后一个数据时,fmt_end信号应置高并保持一个时钟周期。
相邻的数据包之间应至少有一个时钟周期的空闲,即fmt_end变为低电平后,至少经过一个时钟周期fmt_req才可以再次置高。

  根据文档描述,我们可以设计以下握手流程。

formatter握手时序关系

  当前发送任务完毕时,

  先朝下级看,看arbiter有没有发送请求,跟arbiter握手

formatter发送结束之后,就将询问信号fmt_id_req置高。如图所示

在这里插入图片描述

fmt_startfmt_end之间表示正在发送的时候,fmt_id_req_o为高,表示当前formatter空闲。(当然了,这里空出了一个时钟周期,其实也可以不空。因为设计文档说相邻数据包之间间隔至少1个时钟周期,而上下级握手是需要时间的,因此不需要再额外空出一个周期。)

回顾上一章的结构图,fmt_id_req_o是和仲裁信号产生部分相连,当clk_i上升沿到来时发现fmt_id_req_o为1,就会刷新当前综合优先级最高的通道并选通。如下图所示,当fmt_id_req_o为1的下一个周期就会刷新并选通最优先的通道。(必须是就绪的,有包可以发送的)。
当然这一块也有可以改进的地方,就是fmt_id_req_o信号下降沿可以提早一个周期结束,以防止数据发送时优先级突变。不过这里因为假想优先级是一开始在寄存器中设定好的,不考虑数据传输时优先级突然改变的情况。因此仍然保留原来的设计。

在这里插入图片描述

arbiter刷新优先级选通相应通道(也可能所有通道都没就绪,就选通空通道)可以根据arbiter输入的优先通道信号来知道有没有就绪,如果时11空通道就表示没用就绪(这是一个隐藏的握手信号,arbiter向formatter握手表示是否准备发送)

因此置高完fmt_id_req就等待有效通道不是11就行了(这里注意信号不要重叠,先让优先级刷新完再判断,因为fmt_id_req过一个周期才是刷新后的通道,所以判断需要等待一个周期再判断)

不是11代表arbiter就绪,可以发送。这时候已经自动选通了,相当于直连slave,因此slave就等待一个ack信号就可以发送了。现在下级已经就绪,我们通过不是11就向上握手,等待上级进行允许发送。当有效通道不是11,则fmt_req置高,否则继续等待直到不是11.

再朝上级看,formatter等待外部传入允许发送的信号grant

fmt_req置高之后,等待外部允许发送信号fmt_grant,当fmt_grant(为高时,在下一个周期
置低fmt_req
fmt_start输出一个时钟周期的高电平脉冲
fmt_data开始送出第一个数据

[注意]:这里有严格的时序关系,就是grant为高的下个时钟周期就要开始发送第一个数据了,这就要求grant直连的信号slave的ack也满足这样的时序关系,具体而言就是要求slave中的数据在ack之后下一个时钟周期就要发送,valid信号同理
具体做法就是采用了一些简单的波形变换(因为之前的设计还没有涉及到握手时序,波形不满足,所以我又修改了一下,现在slave中波形就满足了时序关系,如下图黄色波形所示)
请添加图片描述

end信号我们在slave中已经产生过了,只需要直接连接过来,通过多路选择器选通即可

握手结束,等待数据发送完毕

然后再fmt_id_req置高开始下一次循环


第二节 fromatter代码实现

  文件名:formatter.v
  具体注释详见代码

/***************<MCDF 整形器 formatter>*********************/
`timescale 1ns/100ps
/***************<端口声明>*********************/
module formatter
(
    input   wire            clk_i,
                            rstn_i,

    input   wire            a2f_val_i,//!!!这个信号没用到,检查是否可以去除
    input   wire    [1:0]   a2f_id_i,
    input   wire    [31:0]  a2f_data_i,
    input   wire    [2:0]   a2f_pkglen_sel_i,

    input   wire            fmt_grant_i,
    
    input   wire            a2f_end_i,//增加的信号,在arbiter中判断end更方便
    
    output  wire            f2a_ack_o,
    output  reg             fmt_id_req_o,

    output  wire    [1:0]   fmt_child_o,
    output  reg     [5:0]   fmt_length_o,
    output  wire            fmt_req_o,
    output  wire    [31:0]  fmt_data_o,
    output  reg             fmt_start_o,
    output  wire            fmt_end_o

);
/***************<中间信号>*********************/
reg fmt_end_d1;//end信号时延一个周期
reg fmt_id_req_o_d1;//fmt_id_req_o信号延迟一个周期,用于波形变换,使得fmt_req上升沿延后
/***************<时序电路>*********************/
//fmt_id_req_o信号延迟一个周期
always @(posedge clk_i or negedge rstn_i) begin
    if(~rstn_i)
        fmt_id_req_o_d1<=0;
    else
        fmt_id_req_o_d1<=fmt_id_req_o;
end
// >>>fmt_end_d1用于产生fmt_id_req_o,并保证两次发送之间的时隙
//end信号时延一个周期
always @(posedge clk_i or negedge rstn_i) begin
    if(~rstn_i)
        fmt_end_d1<=0;
    else
        fmt_end_d1<=fmt_end_o;
end
// >>>fmt_id_req_o<<<
//1产生向下握手信号 fmt_id_req_o,等待优先通道不为11
always @(posedge clk_i or negedge rstn_i) begin
    if(~rstn_i)begin
        fmt_id_req_o<=1;//复位时没有发送占用,因此可以向下级握手
    end
    else if(fmt_end_d1)begin//因为end和start并不会重合,所以谁先谁后判断不影响
        fmt_id_req_o<=1;
    end
    else if(fmt_grant_i)begin
        fmt_id_req_o<=0;
    end
    else
        fmt_id_req_o<=fmt_id_req_o;
end
// >>> fmt_start_o 是f2a_ack_o延迟一个周期
always @(posedge clk_i or negedge rstn_i) begin
    if(~rstn_i)
        fmt_start_o<=0;
    else
        fmt_start_o<=f2a_ack_o;
end
/***************<组合电路>*********************/
// >>>fmt_req_o<<<
//等待优先通道不为11,产生 fmt_req_o,表示下级就绪,当出现 fmt_grant_i 时再置低
//采用 波形变换 方式实现 优先通道不为11&fmt_id_req_o,参考时序参考图
assign  fmt_req_o=fmt_id_req_o & (a2f_id_i!=2'b11) & fmt_id_req_o_d1;
//波形变换,ack信号,在通道联通后,直送给slave
assign  f2a_ack_o=fmt_grant_i & fmt_req_o;
//一些直连信号
assign  fmt_end_o=a2f_end_i;//直连arbiter中的end信号(这个信号是另加的,因为arbiter中判断end更容易)
assign  fmt_data_o=a2f_data_i;
assign  fmt_child_o=a2f_id_i;
always @(*) begin//数据包长度译码(在slave_FIFO里面时用求幂实现)
    case(a2f_pkglen_sel_i) 
        3'b000:fmt_length_o<=6'd4;
        3'b001:fmt_length_o<=6'd8;
        3'b010:fmt_length_o<=6'd16;
        3'b011:fmt_length_o<=6'd32;
        default: 
            fmt_length_o<=6'd32;
    endcase
end

endmodule


第三节 fromattertestbench代码实现

  文件名:formatter_tb.v
  具体注释详见代码

`timescale 1ns/100ps
`include "formatter.v"
/*************************<端口声明>*********************/
module formatter_tb;
reg             clk_i;
reg             rstn_i;
reg             a2f_val_i;
reg     [1:0]   a2f_id_i;
reg     [31:0]  a2f_data_i;
reg     [2:0]   a2f_pkglen_sel_i;
reg             fmt_grant_i;
reg             a2f_end_i;          //增加的

wire            f2a_ack_o;
wire            fmt_id_req_o;
wire    [1:0]   fmt_child_o;
wire    [5:0]   fmt_length_o;
wire            fmt_req_o;
wire    [31:0]  fmt_data_o;
wire            fmt_start_o;
wire            fmt_end_o;
/*************************<实例化>*********************/
formatter fm1(
    clk_i,
    rstn_i,
    a2f_val_i,
    a2f_id_i,
    a2f_data_i,
    a2f_pkglen_sel_i,
    fmt_grant_i,
    a2f_end_i,       
    
    f2a_ack_o,
    fmt_id_req_o,
    fmt_child_o,
    fmt_length_o,
    fmt_req_o,
    fmt_data_o,
    fmt_start_o,
    fmt_end_o
);
/*************************<信号产生>*********************/
initial begin
        clk_i<=0;
        rstn_i<=0;

        a2f_val_i<=0;           //这个信号没用到
        
        a2f_id_i<=2'b11;         //接通的通道
        a2f_data_i<=$random;    //接通通道的数据
        a2f_pkglen_sel_i<=0;    //与a2f_id_i同步变化
        fmt_grant_i<=0;
        a2f_end_i<=0;  
        $dumpfile("formatter.vcd");
        $dumpvars;
#15     rstn_i<=1;
#40 
@(posedge clk_i)    a2f_id_i<=2'b00;
#30
@(posedge clk_i)    fmt_grant_i<=1;
@(posedge clk_i)    fmt_grant_i<=1;
@(posedge clk_i)    fmt_grant_i<=0;
@(posedge clk_i)    fmt_grant_i<=0;
@(posedge clk_i)    a2f_end_i<=1;
@(posedge clk_i)    a2f_end_i<=0;
#500    $finish;
end
always @(posedge clk_i) begin
    a2f_data_i<=$random;
end
always #10 clk_i<=~clk_i;
always @(*) begin//模拟arbiter中的信号同步
    case(a2f_id_i)
        2'b00:a2f_pkglen_sel_i<=3'b000;
        2'b01:a2f_pkglen_sel_i<=3'b001;
        2'b10:a2f_pkglen_sel_i<=3'b010;
        2'b11:a2f_pkglen_sel_i<=3'b000;
    endcase
end
endmodule

  仿真结果如下,同样地,因为涉及到上下级时序配合,因此此处也省略波形分析,将在最后的MCDF顶层仿真中进行详细分析。
请添加图片描述


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

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

相关文章

钡铼技术S274数据遥测终端机

钡铼技术S274数据遥测终端机功能特点&#xff1a; 内置 2 路 DC 直流电源输出&#xff0c;无需单独额外增加变送器的电源适配器&#xff0c;节省布线成本&#xff1b;  采用完备的防掉线机制&#xff0c;保证数据终端永远在线&#xff0c;掉线重发数据以及掉线短信通知用户…

第38篇 网络(八)TCP(二)

导语 在上一节里我们使用TCP服务器发送一个字符串&#xff0c;然后在TCP客户端进行接收。在这一节将重新写一个客户端程序和一个服务器程序&#xff0c;这次实现客户端进行文件的发送&#xff0c;服务器进行文件的接收。有了上一节的基础&#xff0c;这一节的内容就很好理解了…

postgresql_internals-14 学习笔记(三)冻结、rebuild

一、 Freezing 冻结 1. 引入原因 简单说来就是目前pg事务id只有32位&#xff0c;大业务量下很可能用完&#xff0c;触发事务id回卷&#xff08;循环使用&#xff09;。而pg是根据事务id大小判断可见性的&#xff0c;如果新事务却使用了小id&#xff0c;旧事务将可以看到新事务…

win下 conda 虚拟环境没有名字怎么进入

本文主要介绍windows下&#xff0c;在conda 虚拟环境名字消失后的解决办法。主要介绍两种解决方案。 文章目录前言解决方案一&#xff1a;往.condarc文件中添加envs_dirs1. 设置envs_dirs2. 重新查看虚拟环境解决方案二&#xff1a;直接通过path 激活虚拟环境总结前言 我们都知…

Grafana 监控大屏可视化图表

Grafana 系列文章&#xff0c;版本&#xff1a;OOS v9.3.1 Grafana 的介绍和安装Grafana监控大屏配置参数介绍&#xff08;一&#xff09;Grafana监控大屏配置参数介绍&#xff08;二&#xff09;Grafana监控大屏可视化图表 前面我们以Time series 图表为例&#xff0c;学习了面…

每天投递一两个公司,我连续投了三个月

作者&#xff1a;阿秀校招八股文学习网站&#xff1a;https://interviewguide.cn你好&#xff0c;我是阿秀。阿秀以前在秋招的时候投递过八九十份简历&#xff0c;当时还没有简历一键上传功能&#xff0c;很多时候都需要自己去那些公司注册账号&#xff0c;然后找到校园招聘模块…

手机备忘录误删恢复的操作方法

手机备忘录在使用的过程中&#xff0c;会有多种不同的操作&#xff0c;通过不同的操作来实现不同的效果。对于有的内容来说&#xff0c;是可以过期删除的&#xff0c;但是在删除这个操作的过程当中&#xff0c;如果不小心把有用的东西误删了&#xff0c;那么恢复误删内容的操作…

行内块元素因换行带来的间隔问题及解决方法

行内块元素因换行带来的间隔问题 先看一个案例所展示的行内块元素因换行带来的间隔问题,俺直接上截图 再来一张截图可以更加清楚地看见行内块元素之间因换行而带来的间隔 从上方所有图片可以看出,行内块元素之间一行并排放置时编译器中的换行操作会在浏览器渲染时带来行内块元…

鸡汤来了

这几天&#xff0c;网上铺天盖地都是各种感染新冠的消息&#xff0c;连我一直关注的和菜头也感染上了&#xff0c;关键是连怎么感染的都不知道。他写道&#xff1a;我也很委屈。自从北京开始比拼首阳之后&#xff0c;我的确是缩在家里&#xff0c;想着越晚感染越好。为了达到这…

基于STM32单片机和RFID的智能考勤系统设计

提示&#xff1a;记录2022年4月做的毕设 文章目录前言一、任务书1.1设计(研究)目标:1.2设计(研究)内容:二、思路三、硬件3.1RFID模块3.2 时钟模块四、联系我五、代码以及框图等资料喜欢请点赞哦&#xff01;前言 基于STM32的考勤系统&#xff0c;主控使用STM32F103ZET6&#x…

代码随想录算法训练营第四十八天|198.打家劫舍、213.打家劫舍II、337.打家劫舍 III

198.打家劫舍 当前房屋偷与不偷取决于 前一个房屋和前两个房屋是否被偷了。 所以这里就更感觉到&#xff0c;当前状态和前面状态会有一种依赖关系&#xff0c;那么这种依赖关系都是动规的递推公式。 动规五部曲分析如下&#xff1a; 确定dp数组以及下标的含义 dp[i]&#xf…

华为的 HCIE 和 HCIP 的发展趋势如何

华为认证&#xff1a; 华为认证是华为基于“平台生态”战略&#xff0c;围绕“云-管-端”协同的新ICT技术架构&#xff0c;打造的业界覆盖ICT领域最广的认证体系&#xff0c;华为认证覆盖ICT领域&#xff0c;致力于提供领先的人才培养体系和认证标准&#xff0c;培养数字化时代…

基于51单片机的DS12C887电子钟万年历带农历温度

基于51单片机的DS12C887电子钟万年历( proteus仿真程序讲解视频&#xff09; 仿真图proteus 7.8及以上 程序编译器&#xff1a;keil 4/keil 5 编程语言&#xff1a;C语言 设计编号&#xff1a;S0028 资料下载链接&#xff08;可点击&#xff09;&#xff1a; 讲解视频 基…

通信协议综述-第2讲 | 网络分层的真实含义是什么?

长时间从事计算机网络相关的工作,我发现,计算机网络有一个显著的特点,就是这是一个不仅需要背诵,而且特别需要将原理烂熟于胸的学科。很多问题看起来懂了,但是就怕往细里问,一问就发现你懂得没有那么透彻。 我们上一节列了之后要讲的网络协议。这些协议本来没什么稀奇,每…

微服务框架 SpringCloud微服务架构 分布式缓存 41 Redis 持久化 41.2 RDB 持久化【底层原理】

微服务框架 【SpringCloudRabbitMQDockerRedis搜索分布式&#xff0c;系统详解springcloud微服务技术栈课程|黑马程序员Java微服务】 分布式缓存 文章目录微服务框架分布式缓存41 Redis 持久化41.2 RDB 持久化【底层原理】41.2.1 RDB41.2.2 总结41 Redis 持久化 41.2 RDB 持…

【IDEA】解决IDEA插件使用Lombok找不到符号问题

1.概述 今天合并了有冲突的代码,合并完毕后,发现运行的时候找不到符号。类似如下报错 java: 找不到符号* 符号: 方法 setData(java.lang.String)* 位置: 类型为com.fastjson.demo.lombok.

Vue渲染器(三):简单diff算法

渲染器&#xff08;三&#xff09;&#xff1a;简单diff算法 我们将介绍渲染器的核心Diff算法。简单来说就是当新旧vnode的子节点都是一组节点时&#xff0c;为了以最小的性能开销完成更新操作&#xff0c;需要比较两组子节点。 1.减少DOM操作的性能开销&#xff1a; 核心Di…

第46篇 进阶(六) 国际化

导语 在第2篇中讲述如何显示中文时&#xff0c;曾提到使用QTextCodec和tr()的方式直接显示中文&#xff0c;其实这只是一种临时的方法&#xff0c;方便我们快速完成程序&#xff0c;显示效果。当真正要发布一个程序时&#xff0c;最好的方式是在程序中使用英文字符串&#xff…

四、函数基础、函数种类、形实参和映射关系

四、函数基础、函数种类、形实参和映射关系 编程基本原则&#xff1a;高内聚、低耦合。 我总结&#xff1a; 模块自身&#xff0c;要求高内聚&#xff1a;模块中代码相关性变强&#xff0c;代码紧密联系程度变高&#xff0c;希望它能独立完成一个功能。模块之间&#xff0c;…

【20221213】【每日一题】零钱兑换II

给你一个整数数组 coins 表示不同面额的硬币&#xff0c;另给一个整数 amount 表示总金额。 请你计算并返回可以凑成总金额的硬币组合数。如果任何硬币组合都无法凑出总金额&#xff0c;返回 0 。 假设每一种面额的硬币有无限个。 题目数据保证结果符合 32 位带符号整数。 …