从零开始 verilog 以太网交换机(二)Mac接收控制器的设计与实现 🔈声明: 😃博主主页:王_嘻嘻的CSDN主页 🧨 从零开始 veril
🔈声明:
😃博主主页:王_嘻嘻的CSDN主页
🧨 从零开始 verilog 以太网交换机系列专栏:点击这里
🔑未经作者允许,禁止转载,侵权必删
🚩关注本专题的朋友们可以收获一个经典交换机设计的全流程,包括设计与验证(FPGA);以太网MAC的基础知识。新手朋友们还将获得一个具有竞争力的项目经历,后续整个工程和代码下载链接也都会放在csdn和公众号内
本章进行MAC控制器的设计与实现,共分为两个部分:接收控制器和发送控制器。整体架构可以参考:从零开始 verilog 以太网交换机(一)架构分析,本文将首先分析MAC接收控制器的设计。
MAC控制器负责以太网的MAC层和PHY层之间的数据交换,主要是根据MII标准接口规范进行数据收发。
对于本章的MAC接收控制器而言,其功能包括以下4点:
MAC接收控制器一端连接标准MII(Media Independent Interface,媒体独立接口)①接口,另一端连接交换机内部的数据帧处理单元。因为交换机的系统时钟 clk 和PHY的 rx_clk 为异步时钟,且而后级的数据处理单元最坏情况需要接收4 Port的数据帧传输,可能出现满载,来不及处理接收的数据,所以在MAC接收器内部,我们设计了异步FIFO,以完成数据缓冲的作用,另一方面FIFO也可以完成握手信号的逻辑,对MAC控制器内部的状态机进行反压②。
所以,MAC接收控制器的输入为MII接口,输出为FIFO接口形式,由于FIFO比较基础,下面就仅罗列MII接口。
①:MII接口为PHY<->MAC间接口的统称,本工程用了最原始的MII接口,随着速率要求还有GMII、SGMII、RGMII等多类MII接口,感兴趣的朋友可以自行了解。
②:复杂的硬件设计中模块间通常存在请求(request)和响应(response)的握手机制,当后级模块内部来不及处理数据时,会通知前级模块,禁止前级继续发送数据,并以此类推,直到数据流源头被暂停,这个行为就叫做反压。
根据以太网MAC层的规范,控制器还需要对数据帧进行CRC-32校验,由于CRC(Cyclic redundancy check,循环冗余校验)本质就是反馈多项式,和LFSR(linear feedback shift reGISter,线性反馈移位寄存器)类似,在本专题就不展开介绍了,需要了解的朋友可以自行搜索学习,通常CRC的多项式公式可以直接通过生成器来生成。(CRC生成器将放在本号的资源栏中,有需要的可以下载,或者关注博主的公众号,也有下载链接)
综合上述FIFO和CRC的需求,最后MAC接收控制器的整体架构图如下:
MAC接收控制器中有两个FIFO,我们将存放数据的FIFO称作data_fifo,将存放状态的称作state_fifo。
由于在发送端每个数据帧的末尾会插入CRC-32的计算值,这会使得在接收端每一个正确的数据帧的CRC校验值为一固定值,当不为此值时,即代表发生CRC error。在设计中,我们可以将该固定值设置为一个parameter,以便大家使用不同的多项式实现CRC。
此外,我们通过检查数据帧的起始标志和结束标志,来判断帧边界,并不断记录数据帧的字节数,当到达结束标志后,检查数据长度是否合规,数据是否是字节对齐的,若发生错误则在state_fifo中记录,这样就完成了MAC接收控制器的所有功能。
需要注意的是数据帧的开始符按照MAC帧格式来看是0x10101011,其中前6位的101010是MAC前导符的延申。前导符是比特‘10’的集合,用于PHY serdes③(串行器)做时钟恢复用(因为serdes的时钟和数据合并在一根数据线上,所以接收端需要做时钟恢复)。所以接收到0x1011即代表接收到有效数据帧(SFD,start frame describe)。
另一点是MAC帧要求在帧与帧之间必须存在间隔,即一个数据帧发送完毕后rx_dv必定存在低电平时刻,所以在设计中,我们使用一个状态机来表示数据帧处于的不同状态,并用于控制各个变量,状态跳转图如下:
将状态机共分为4个状态:ST_SOF、ST_WaiT、ST_EOF、ST_DONE,分别对应等待MAC帧开始、等待帧起始符、等待MAC结束、将各信息传递至fifo。
③:serdes是一种将并行数据转换成高速串行数据发送的技术,通常是数模混合设计,有相当多优点,所以也十分复杂。当前多数主流的物理接口都用到了serdes,包括PCI-e,以太网等。
控制器的设计并不复杂,Verilog代码将放在下面,Testbench就不展示了,有需要的可以等专题结束后在资源中下载,或者去我的公众号获得链接。
module mac_r(//system interfaceinput clk,input rst_n,//MII interfaceinput rx_clk,input rx_dv,input [3:0] rx_d,//mac-r - interface muxinput data_fifo_rd,output [7:0] data_fifo_dout,input state_fifo_rd,output [15:0] state_fifo_dout,output state_fifo_empty ); parameter CRC_RESULT = 32'hc704dd7b;parameter ST_SOF = 4'b0001;parameter ST_WAIT = 4'b0010;parameter ST_EOF = 4'b0100; parameter ST_DONE = 4'b1000;parameter BCNT_MAX = 1518;parameter BCNT_MIN = 64;reg rx_dv_dly0; //用于输入数据有效的采样 reg rx_dv_dly1;reg [3:0] rx_d_dly0;reg [3:0] rx_d_dly1;//信号有效信号的上升沿表示帧开始,下降沿则表示帧结束wire sof; //start of frame 帧开始标志 只能表示有MAC帧来了wire eof; //end of frame 帧结束标志wire sfd; //start frame decribe 帧起始符 sof不代表真正数据帧的开始,sdf才代表!wire mac_r_rdy; //表示mac_r当前有接收能力//控制状态机reg [3:0] cur_state;reg [3:0] next_state;//byte cnt变量wire bcnt_clr; //清零信号reg [11:0] bcnt;reg frame_vld; //数据帧有效信号//fifowire [7:0] data_fifo_din; wire data_fifo_wr;wire [11:0] data_fifo_wr_cnt;reg [15:0] state_fifo_din;reg state_fifo_wr;wire state_fifo_full;wire [31:0] crc_reg;always @(posedge rx_clk or negedge rst_n)begin if(!rst_n)begin rx_dv_dly0 <= 1'b0; rx_dv_dly1 <= 1'b0; rx_d_dly0[3:0] <= 4'b0; rx_d_dly1[3:0] <= 4'b0; end else begin rx_dv_dly0 <= rx_dv; rx_dv_dly1 <= rx_dv_dly0; rx_d_dly0[3:0] <= rx_d[3:0]; rx_d_dly1[3:0] <= rx_d_dly0[3:0]; endend assign sof = !rx_dv_dly1 & rx_dv_dly0;assign eof = rx_dv_dly1 & !rx_dv_dly0;assign sfd = rx_dv_dly0 & (rx_d_dly0[3:0]==4'b1011);//三段式状态机always @(posedge rx_clk or negedge rst_n)begin if(!rst_n) cur_state[3:0] <= ST_SOF; else cur_state[3:0] <= next_state[3:0];endalways @(*)begin case(cur_state[3:0]) ST_SOF: next_state[3:0] = (sof & mac_r_rdy) ? !sfd ? ST_WAIT : ST_EOF : ST_SOF; ST_WAIT: next_state[3:0] = rx_dv_dly0 ? sfd ? ST_EOF : ST_WAIT : ST_SOF; ST_EOF: next_state[3:0] = eof ? ST_DONE : ST_EOF; ST_DONE: next_state[3:0] = ST_SOF; default: next_state[3:0] = ST_SOF; endcase endassign bcnt_clr = (sof & sfd) | (cur_state[3:0]==ST_WAIT & sfd);always @(posedge rx_clk or negedge rst_n)begin if(!rst_n) bcnt[11:0] <= 12'b0; else if(bcnt_clr) bcnt[11:0] <= 12'b0; else bcnt[11:0] <= bcnt[11:0] + 12'b1;endalways @(posedge rx_clk or negedge rst_n)begin if(!rst_n) frame_vld <= 1'b0; else if(cur_state[3:0]==ST_EOF & eof) frame_vld <= 1'b0; else if( (cur_state[3:0]==ST_SOF & sof & sfd) | (cur_state[3:0]==ST_WAIT & rx_dv_dly0 & sfd)) frame_vld <= 1'b1;endassign data_fifo_wr = frame_vld & bcnt[0] & rx_dv_dly0; //当存够两个rx_data时,且帧有效时,向data_fifo中存一次数据assign data_fifo_din = {rx_d_dly0,rx_d_dly1};assign mac_r_rdy = (data_fifo_wr_cnt[11:0]> 4096 - BCNT_MAX);always @(posedge rx_clk or negedge rst_n)begin if(!rst_n) state_fifo_wr <= 1'b0; else if(cur_state[3:0]==ST_EOF & eof) state_fifo_wr <= 1'b1; else state_fifo_wr <= 1'b0;endalways @(posedge rx_clk or negedge rst_n)begin if(!rst_n) state_fifo_din[15:0] <= 16'b0; else if(cur_state[3:0]==ST_EOF & eof)begin state_fifo_din[10:0] <= bcnt[11:1]; state_fifo_din[14] <=( (bcnt[11:1] > BCNT_MAX) | (bcnt[11:1] < BCNT_MIN) | bcnt[0] ) ? 1'b1 :1'b0; state_fifo_din[15] <= (crc_reg[31:0]==CRC_RESULT) ? 1'b0 : 1'b1; endenddata_fifo x_data_fifo( .rst(~rst_n), .wr_clk(rx_clk), .rd_clk(clk), .din(data_fifo_din[7:0]), .wr_en(data_fifo_wr), .rd_en(data_fifo_rd), .dout(data_fifo_dout[7:0]), .full(), .empty(), .rd_data_count(), .wr_data_count(data_fifo_wr_cnt[11:0]) ); state_fifo x_state_fifo( .rst(~rst_n), .wr_clk(rx_clk), .rd_clk(clk), .din(state_fifo_din[15:0]), .wr_en(state_fifo_wr), .rd_en(state_fifo_rd), .dout(state_fifo_dout[15:0]), .full(state_fifo_full), .empty(state_fifo_empty) ); crc32 x_crc32( .clk(clk), .rst_n(rst_n), .data(data_fifo_din[7:0]), .init(sof), .cal(data_fifo_wr), .vld(data_fifo_wr), .crc_reg(crc_reg[31:0]), .crc() //大端输出 );endmodule
来源地址:https://blog.csdn.net/sz_woshishazi/article/details/128540180
--结束END--
本文标题: 从零开始 verilog 以太网交换机(二)MAC接收控制器的设计与实现
本文链接: https://lsjlt.com/news/401533.html(转载时请注明来源链接)
有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341
2024-01-21
2023-10-28
2023-10-28
2023-10-27
2023-10-27
2023-10-27
2023-10-27
回答
回答
回答
回答
回答
回答
回答
回答
回答
回答
0