-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathpcs_tx.v
226 lines (199 loc) · 6.34 KB
/
pcs_tx.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
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
/* 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. */
/* PCS on the transmission path
*
* This module does not support a configurable data width,
* it expects 256b of data from the mac every cycle.
*
* The meaning of "lane" is difference for 40g than 10g.
* In 10g each block had 2x4 lanes, in 40g the xgmii data
* is composed of 4 lanes of 1 block width. */
module pcs_tx#(
parameter IS_10G = 0,
parameter IS_TB = 1,
parameter LANE_N = IS_10G ? 1 : 4,
parameter DATA_W = 64,
parameter HEAD_W = 2,
/* verilator lint_off UNUSEDPARAM */
parameter BLOCK_W = DATA_W+HEAD_W,
/* verilator lint_on UNUSEDPARAM */
parameter KEEP_W = DATA_W/8,
parameter LANE0_CNT_N = IS_10G ? 2 : 1,
parameter XGMII_DATA_W = LANE_N*DATA_W,
parameter XGMII_KEEP_W = LANE_N*KEEP_W
)(
input pcs_clk, /* pcs common clk */
input [LANE_N-1:0] tx_par_clk, /* serdes parallel clk */
input nreset,
// MAC
input [LANE_N-1:0] ctrl_v_i,
input [LANE_N-1:0] idle_v_i,
input [LANE_N*LANE0_CNT_N-1:0] start_v_i,
input [LANE_N-1:0] term_v_i,
input [LANE_N-1:0] err_v_i,
input [XGMII_DATA_W-1:0] data_i, // tx data
input [XGMII_KEEP_W-1:0] keep_i,
output marker_v_o,// alignement marker added this cycle,not used in 10GBASE-R
output ready_o,// gearbox accept next data
/* SerDes */
output [LANE_N*DATA_W-1:0] serdes_data_o
);
// encoder
logic scram_v;
logic [LANE_N-1:0] unused_enc_head_v;
logic [XGMII_DATA_W-1:0] data_enc; // encoded
// scrambler
logic [XGMII_DATA_W-1:0] data_scram; // scrambled
// sync header is allways valid
logic [LANE_N*HEAD_W-1:0] sync_head;
// gearbox
/* gearbox full has the same value every gearbox
* regardless of the lane, we can ignore all of them
* but 1 as long as we are sending all the data blocks
* within the same cycle, this may be changed in future
* versions */
/*verilator lint_off UNUSEDSIGNAL */
logic [LANE_N-1:0] gb_accept;
/*verilator lint_on UNUSEDSIGNAL */
/* input to gearbox */
logic [LANE_N-1:0] gb_nreset;
logic [LANE_N*HEAD_W-1:0] gb_head;
logic [LANE_N*DATA_W-1:0] gb_data;
genvar l;
generate
for( l = 0; l < LANE_N; l++ ) begin : gen_enc_lane
// encode
pcs_enc_lite #(.DATA_W(DATA_W), .IS_10G(IS_10G))
m_pcs_enc(
.ctrl_v_i(ctrl_v_i[l]),
.idle_v_i(idle_v_i[l]),
.start_v_i(start_v_i[l*LANE0_CNT_N+LANE0_CNT_N-1:l*LANE0_CNT_N]),
.term_v_i(term_v_i[l]),
.err_v_i(err_v_i[l]),
.part_i('0),
.data_i(data_i[l*DATA_W+DATA_W-1:l*DATA_W]), // tx data
.keep_i(keep_i[l*KEEP_W+KEEP_W-1:l*KEEP_W]),
.keep_next_i('X),
.head_v_o(unused_enc_head_v[l]),
.sync_head_o(sync_head[l*HEAD_W+HEAD_W-1:l*HEAD_W]),
.data_o(data_enc[l*DATA_W+DATA_W-1:l*DATA_W])
);
end
endgenerate
// scramble
_64b66b_tx #(.LEN(XGMII_DATA_W))
m_64b66b_tx(
.clk(pcs_clk),
.nreset(nreset),
.valid_i(scram_v),
.data_i (data_enc ),
.scram_o(data_scram)
);
if ( !IS_10G ) begin : gen_not_10g
/* alignement marker insertion */
logic marker_v;
logic [XGMII_DATA_W-1:0] data_mark;
logic [LANE_N*HEAD_W-1:0] sync_head_mark;
am_tx #(.LANE_N(LANE_N), .HEAD_W( HEAD_W ), .DATA_W(DATA_W))
m_align_market(
.clk(pcs_clk),
.nreset(nreset),
.valid_i(gb_accept[0]),
.head_i(sync_head),
.data_i(data_scram ),
.marker_v_o(marker_v),
.head_o(sync_head_mark ),
.data_o(data_mark )
);
if (!IS_TB) begin: gen_cdc_fifo
/* CDC
* alignement marker -> cdc -> gearbox */
logic [LANE_N-1:0] rd_cdc_req;
logic [LANE_N-1:0] rd_cdc_empty;
logic [LANE_N-1:0] wr_cdc_valid;
logic [BLOCK_W-1:0] rd_cdc_data[LANE_N-1:0];
logic [BLOCK_W-1:0] wr_cdc_data[LANE_N-1:0];
for(l=0; l<LANE_N;l++) begin: gen_cdc_lane
/* wr */
assign wr_cdc_valid[l] = ~nreset;
assign wr_cdc_data[l] = {data_mark[l*DATA_W+:DATA_W],
sync_head_mark[l*HEAD_W+:HEAD_W]};
/* rd
* req : only gearbox accept a new data
* empty : fifo should never be empty when gearbox can accept
* new data, this only happens during reset, using this
* as sync reset confition */
assign rd_cdc_req[l] = gb_accept[l];
assign gb_nreset[l] = rd_cdc_req[l] & rd_cdc_empty[l];
cdc_fifo m_cdc_fifo_tx (
.wrclk(pcs_clk),
.rdclk(tx_par_clk[l]),
/* wr */
.wrreq(wr_cdc_valid[l]),
.data(wr_cdc_data[l]),
/* rd */
.q(rd_cdc_data[l]),
.rdreq(rd_cdc_req[l]),
.rdempty(rd_cdc_empty[l])
);
assign gb_data[l*DATA_W+:DATA_W] = rd_cdc_data[l][BLOCK_W-1:HEAD_W];
assign gb_head[l*HEAD_W+:HEAD_W] = rd_cdc_data[l][HEAD_W-1:0];
end // gen_cdc_lane
end else begin : gen_no_cdc
// gearbox data : marked data
assign gb_nreset = {LANE_N{nreset}};
assign gb_data = data_mark;
assign gb_head = sync_head_mark;
end //!IS_TB
// scrambler, marker, data ready
assign scram_v = ~marker_v;
assign marker_v_o = marker_v;
assign ready_o = ~marker_v;
end else begin : gen_10g
// gearbox data : scrambled data
assign gb_data = data_scram;
assign gb_head = sync_head;
assign gb_nreset = nreset;
// scrambler
assign scram_v = gb_accept[0];
/* PCS is non blocking in 10G mode
* The only case where the PCS becomes blocking is when we
* are adding the alignement marker.
* And, as there is no alignement marker for 10GBASE and this
* signal is not expected to be used in this configuration */
assign marker_v_o = 1'bX;
assign ready_o = gb_accept[0];
end //!IS_10G
/* gearbox */
generate
for(l=0; l<LANE_N; l++) begin : gen_gearbox_lane
gearbox_tx #(
.DATA_W(DATA_W),
.HEAD_W(HEAD_W)
)m_gearbox_tx(
.clk(tx_par_clk[l]),
.nreset(gb_nreset[l]),
.head_i(gb_head[l*HEAD_W+HEAD_W-1:l*HEAD_W]),
.data_i(gb_data[l*DATA_W+DATA_W-1:l*DATA_W]),
.accept_v_o(gb_accept[l]),
.data_o(serdes_data_o[l*DATA_W+DATA_W-1:l*DATA_W])
);
end
endgenerate
`ifdef FORMAL
always @(posedge pcs_clk) begin
// gearbox state should be the same regardless of the lane
// when we send all the data within the same cycle
// note : only applies when there are no cdc's between gearbox
if ( CNT_N == 1 && IS_TB == 0 ) begin
for(l=0; l<LANE_N; l++) begin
assert ( gearbox_full[0] == gearbox_full[l]);
end
end
end
`endif
endmodule