forked from Essenceia/ethernet-physical-layer
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathxgmii_dec_intf_rx.v
97 lines (80 loc) · 3.17 KB
/
xgmii_dec_intf_rx.v
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
/* Copyright (c) 2023, Julia Desmazes. All rights reserved.
*
* This work is licensed under the Creative Commons Attribution-NonCommercial
* 4.0 International License.
*
* This code is provided "as is" without any express or implied warranties. */
/* XGMII and XLGMII decoder interface, used on rx*/
module xgmii_dec_intf_rx #(
parameter IS_40G = 1, // are comunicating with XLGMII
parameter XGMII_DATA_W = 64,
parameter XGMII_CTRL_W = XGMII_DATA_W/8,
parameter LANE0_CNT_N = IS_40G ? 1 : 2,
parameter DATA_W = 64,
parameter KEEP_W = DATA_W/8,
parameter CTRL_W = 8
)(
// output of decoder
input ctrl_v_i,
input idle_v_i,
input [LANE0_CNT_N-1:0] start_v_i,
input term_v_i,
input err_v_i,
/* verilator lint_off UNUSEDSIGNAL*/
// TODO : Add support for order
input ord_v_i,
/* verilator lint_on UNUSEDSIGNAL*/
input [DATA_W-1:0] data_i, // x(l)gmii data
input [KEEP_W-1:0] keep_i,
output [XGMII_DATA_W-1:0] xgmii_txd_o,
output [XGMII_CTRL_W-1:0] xgmii_txc_o
);
localparam [CTRL_W-1:0]
XGMII_CTRL_IDLE = 8'h07,
XGMII_CTRL_START = 8'hfb,
XGMII_CTRL_TERM = 8'hfd,
XGMII_CTRL_ERR = 8'hfe;
// re-translate term to onehot, we undo modifs done in the
// dec as our primary use case don't target the x(l)gmii
logic [XGMII_CTRL_W-1:0] term_onehot;
logic [XGMII_CTRL_W-1:0] term_onehot_lite;
logic unused_term_onehot_lite_of;
assign { unused_term_onehot_lite_of, term_onehot_lite } = keep_i
+ {{XGMII_CTRL_W-1{1'b0}},1'd1};// + 1
assign term_onehot = {XGMII_CTRL_W{term_v_i}} & term_onehot_lite;
// xgmii control codes
logic [CTRL_W-1:0] ctrl_code;
// idle ctrl will be handled appart
assign ctrl_code = {CTRL_W{|start_v_i}} & XGMII_CTRL_START
| {CTRL_W{term_v_i}} & XGMII_CTRL_TERM
| {CTRL_W{err_v_i}} & XGMII_CTRL_ERR;
logic [XGMII_CTRL_W-1:0] ctrl_lane_v;
logic [XGMII_CTRL_W-1:0] idle_lane_v;
// on idle all bytes should be set to idle ctrl and set all bytes after term to idle
assign idle_lane_v = {XGMII_CTRL_W{ctrl_v_i}} & ({XGMII_CTRL_W{idle_v_i}} | ( {XGMII_CTRL_W{term_v_i}} & ~keep_i ));
assign ctrl_lane_v[0] = ctrl_v_i & ( start_v_i[0] | err_v_i | term_onehot[0]);
// When deleting /I/s, the first four characters after a /T/ shall not be deleted.
genvar i;
generate
for(i=1; i< XGMII_CTRL_W; i++) begin
if ( !IS_40G && i == 3 ) begin
assign ctrl_lane_v[i] = ctrl_v_i & (start_v_i[1] | term_onehot[i]);
end else begin
assign ctrl_lane_v[i] = ctrl_v_i & term_onehot[i];
end
end
endgenerate
// shift output data, term character is moved from first byte in the pcs to
// the end of the valid data in the xgmii
logic [DATA_W-1:0] data_shift;
assign data_shift = term_v_i & ctrl_v_i ? { {CTRL_W{1'bx}}, data_i[DATA_W-1:CTRL_W] } : data_i;
generate
for(i=0; i< XGMII_CTRL_W; i++) begin
assign xgmii_txd_o[i*CTRL_W+CTRL_W-1:i*CTRL_W] = ctrl_lane_v[i] ? ctrl_code
: idle_lane_v[i] ? XGMII_CTRL_IDLE
: data_shift[i*CTRL_W+CTRL_W-1:i*CTRL_W];
end
endgenerate
assign xgmii_txc_o = ctrl_lane_v | idle_lane_v;
// overwrite
endmodule