forked from Essenceia/ethernet-physical-layer
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathxgmii_pcs_10g_enc_intf.v
104 lines (88 loc) · 2.92 KB
/
xgmii_pcs_10g_enc_intf.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
98
99
100
101
102
103
104
/* 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. */
/* interface between the 64b data wide xgmii interface and the pcs */
module xgmii_pcs_10g_enc_intf #(
parameter XGMII_DATA_W = 64,
parameter XGMII_KEEP_W = $clog2(XGMII_DATA_W),
parameter XGMII_CTRL_W = 8,
parameter BLOCK_W = 64,
parameter HEAD_W = 2, // sync header
parameter LANE0_CNT_N = BLOCK_W/( 4 * 8),
parameter LANE0_CNT_W = $clog2(LANE0_CNT_N+1)
)(
input clk,
input nreset,
input [XGMII_DATA_W-1:0] xgmii_txd_i,
input [XGMII_CTRL_W-1:0] xgmii_txc_i,
output ctrl_v_o,
output idle_v_o,
output [LANE0_CNT_W-1:0] start_v_o,
output term_v_o,
output err_v_o,
output [XGMII_KEEP_W-1:0] keep_o // data keep
);
localparam [XGMII_CTRL_W-1:0]
XGMII_CTRL_IDLE = 8'h07,
XGMII_CTRL_START = 8'hfb,
XGMII_CTRL_TERM = 8'hfd,
XGMII_CTRL_ERR = 8'hfe;
// translate xgmii control byte into individual signals
assign ctrl_v_o = |xgmii_txc_i;
// EOF
// eof can be on any xgmii lane, but is the first control position
// get the first ctrl bit
logic [XGMII_CTRL_W-1:0] lsb_ctrl_mask;
logic lsb_ctrl_mask_overflow;
logic [XGMII_CTRL_W-1:0] lsb_ctrl;
logic term_v_lite;
assign {lsb_ctrl_mask_overflow, lsb_ctrl_mask} = ~xgmii_txc_i + {XGMII_CTRL_W-1{1'b0} , 1'b1};
always_comb begin
for( int i=0; i < XGMII_CTRL_W; i++) begin
if( lsb_ctrl_mask[i] ) lsb_ctrl = xgmii_txd_i[i*8+7:i*8];
end
end
assign term_v_lite = lsb_ctrl == XGMII_CTRL_TERM;
// data mask for end of block
logic [XGMII_KEEP_W-1:0] term_keep;
assign term_keep = ~xgmii_txc_i;
// SOF
// sof only present on lane 0
logic has_data;
logic [XGMII_CTRL_W-1:0] msb_lane0_ctrl;
logic [LANE0_CNT_N-1:0] lane0_start_v;
logic [LANE0_CNT_N-1:0] start_v_lite;
assign has_data = ~xgmii_txc_i[XGMII_CTRL_W-1];
genvar i;
generate
for(i=0; i<LANE0_CNT_N; i++) begin
assign lane0_start_v[i] = xgmii_txd_i[i*4*8+7:i*4*8] == XGMII_CTRL_START;
end
endgenerate
assign start_v_lite[0] = lane0_start_v[0] & ~lane0_start_v[1];
assign start_v_lite[1] = lane0_start_v[1];
// ERR
// TODO : Check the rules on error lane placement ?
logic err_v_lite;
assign err_v_lite = lsb_ctrl == XGMII_CTRL_ERR;
// IDLE
logic idle_v_lite;
assign idle_v_lite = lsb_ctrl == XGMII_CTRL_IDLE;
// Validity signals
assign start_v_o = start_v_lite & {LANE0_CNT_N{has_data}};
assign term_v_o = term_v_lite & ~has_data;
assign idle_v_o = idle_v_lite & ~has_data;
assign err_v_o = err_v_lite & ~has_data;
`ifdef FORMAL
logic data_v_f;
assign data_v_f = start_v_o | term_v_o;
always_comb begin
// xcheck
sva_xcheck_xgmii_ctr : assert( ~$isunknown(xgmii_txc_i));
sva_xcheck_xgmii_keep : assert(~data_v_f | data_v_f & ~$isunknown(keep_o));
end
`endif
endmodule