@@ -66,6 +66,18 @@ module axi_lite_dw_converter #(
66
66
parameter int unsigned AxiSlvPortDataWidth = 32'd0 ,
67
67
// / AXI4-Lite data width of the master port.
68
68
parameter int unsigned AxiMstPortDataWidth = 32'd0 ,
69
+ // / Whether to read a transaction size from the AR user bits
70
+ parameter bit UserArSize = 1'b0 ,
71
+ // / Whether to read a transaction size from the AW user bits
72
+ parameter bit UserAwSize = 1'b0 ,
73
+ // / Least significant bit (LSB) of size in AR user fields
74
+ parameter int unsigned UserArSizeLsb = 32'd0 ,
75
+ // / Least significant bit (LSB) of size in AW user fields
76
+ parameter int unsigned UserAwSizeLsb = 32'd0 ,
77
+ // / Assuming AW size in user field, maximum number of inflight reads.
78
+ parameter int unsigned UserArMaxTxns = 32'd0 ,
79
+ // / Assuming AW size in user field, maximum number of inflight writes.
80
+ parameter int unsigned UserAwMaxTxns = 32'd0 ,
69
81
// / AXI4-Lite AW channel struct type. This is for both ports the same.
70
82
parameter type axi_lite_aw_t = logic ,
71
83
// / AXI4-Lite W channel struct type of the slave port.
@@ -132,6 +144,7 @@ module axi_lite_dw_converter #(
132
144
// Input spill register of the AW channel.
133
145
axi_lite_aw_t aw_chan_spill;
134
146
logic aw_chan_spill_valid, aw_chan_spill_ready;
147
+ logic w_progress, aw_progress, b_progress;
135
148
136
149
spill_register # (
137
150
.T ( axi_lite_aw_t ),
@@ -147,19 +160,35 @@ module axi_lite_dw_converter #(
147
160
.data_o ( aw_chan_spill )
148
161
);
149
162
150
- sel_t aw_sel_q, aw_sel_d;
163
+ sel_t aw_sel_q, aw_sel_d, aw_sel_out ;
151
164
logic aw_sel_load;
152
165
// AW channel output assignment
153
166
always_comb begin : proc_aw_chan_oup
154
167
mst_req_o.aw = aw_chan_spill;
155
- mst_req_o.aw.addr = out_address (aw_chan_spill.addr, aw_sel_q);
168
+ mst_req_o.aw.addr = out_address (aw_chan_spill.addr, aw_sel_out);
169
+ end
170
+
171
+ assign aw_progress = aw_chan_spill_valid & mst_res_i.aw_ready;
172
+
173
+ if (UserAwSize) begin : gen_user_aw
174
+ // Couple AW, W, and B FIFO and make requests selectively
175
+ sel_t aw_sel_end, aw_sel_base;
176
+ assign aw_sel_end = UserAwSize ?
177
+ (sel_t ' (1 ) << aw_chan_spill.user[UserAwSizeLsb: + axi_pkg :: SizeWidth]) >> SelOffset : '0 ;
178
+ assign aw_sel_base = aw_chan_spill.aw.addr[SelOffset+ : SelWidth] & sel_t ' (aw_sel_end - 1 );
179
+ assign aw_sel_out = aw_sel_q + aw_sel_base;
180
+ assign mst_req_o.aw_valid = w_progress & b_progress & aw_chan_spill_valid;
181
+ assign aw_chan_spill_ready = w_progress & b_progress & mst_res_i.aw_ready &
182
+ ((& aw_sel_q) | (aw_sel_d == aw_sel_end));
183
+ end else begin : gen_no_user_aw
184
+ // AW, W, and B are uncoupled
185
+ assign aw_sel_out = aw_sel_q;
186
+ assign mst_req_o.aw_valid = aw_chan_spill_valid;
187
+ assign aw_chan_spill_ready = mst_res_i.aw_ready & (& aw_sel_q);
156
188
end
157
- // Slave port aw is valid, if there is something in the spill register.
158
- assign mst_req_o.aw_valid = aw_chan_spill_valid;
159
- assign aw_chan_spill_ready = mst_res_i.aw_ready & (& aw_sel_q);
160
189
161
190
assign aw_sel_load = mst_req_o.aw_valid & mst_res_i.aw_ready;
162
- assign aw_sel_d = sel_t ' (aw_sel_q + 1'b1 );
191
+ assign aw_sel_d = sel_t ' (aw_sel_load ? '0 : aw_sel_q + 1'b1 );
163
192
`FFLARN (aw_sel_q, aw_sel_d, aw_sel_load, '0 , clk_i, rst_ni)
164
193
165
194
// Input spill register of the W channel.
@@ -180,41 +209,82 @@ module axi_lite_dw_converter #(
180
209
);
181
210
182
211
// Data multiplexer on the W channel
183
- sel_t w_sel_q, w_sel_d;
184
- logic w_sel_load;
212
+ sel_t w_sel;
185
213
// W channel output assignment
186
214
assign mst_req_o.w = axi_lite_mst_w_t'{
187
- data: w_chan_spill.data[w_sel_q * AxiMstPortDataWidth+ : AxiMstPortDataWidth],
188
- strb: w_chan_spill.strb[w_sel_q * AxiMstPortStrbWidth+ : AxiMstPortStrbWidth],
215
+ data: w_chan_spill.data[w_sel * AxiMstPortDataWidth+ : AxiMstPortDataWidth],
216
+ strb: w_chan_spill.strb[w_sel * AxiMstPortStrbWidth+ : AxiMstPortStrbWidth],
189
217
default : '0
190
218
} ;
191
- assign mst_req_o.w_valid = w_chan_spill_valid;
192
- assign w_chan_spill_ready = mst_res_i.w_ready & (& w_sel_q);
193
219
194
- assign w_sel_load = mst_req_o.w_valid & mst_res_i.w_ready;
195
- assign w_sel_d = sel_t ' (w_sel_q + 1'b1 );
196
- `FFLARN (w_sel_q, w_sel_d, w_sel_load, '0 , clk_i, rst_ni)
220
+ assign w_progress = w_chan_spill_valid & mst_res_i.w_ready;
221
+
222
+ if (UserAwSize) begin : gen_user_aw_w
223
+ // We must couple the AW, W, and B FIFO; adopt AW channel counts here
224
+ assign mst_req_o.w_valid = aw_progress & b_progress & w_chan_spill_valid;
225
+ assign w_chan_spill_ready = aw_progress & b_progress & mst_res_i.w_ready;
226
+ assign w_sel = aw_sel_out;
227
+ end else begin : gen_no_user_aw_w
228
+ // The W channel can operate uncoupled
229
+ sel_t w_sel_q, w_sel_d;
230
+ logic w_sel_load;
231
+ assign w_sel_load = mst_req_o.w_valid & mst_res_i.w_ready;
232
+ assign w_sel_d = sel_t ' (w_sel_q + 1'b1 );
233
+ `FFLARN (w_sel_q, w_sel_d, w_sel_load, '0 , clk_i, rst_ni)
234
+ assign mst_req_o.w_valid = w_chan_spill_valid;
235
+ assign w_chan_spill_ready = mst_res_i.w_ready & (& w_sel_q);
236
+ assign w_sel = w_sel_q;
237
+ end
197
238
198
239
// B response aggregation
199
240
// Slave port B output is the aggregated error of the last few B responses.
200
241
sel_t b_sel_q, b_sel_d;
201
242
axi_pkg :: resp_t b_resp_q, b_resp_d;
202
243
logic b_resp_load;
244
+ logic b_end;
203
245
204
246
assign slv_res_o.b = axi_lite_b_t'{
205
247
resp: b_resp_q | mst_res_i.b.resp,
206
248
default : '0
207
249
} ;
250
+
251
+ if (UserAwSize) begin : gen_user_aw_b
252
+ // When an upstream AW/W pair completes, store the expected downstream B count
253
+ sel_t b_out;
254
+ stream_fifo # (
255
+ .FALL_THROUGH ( 1'b0 ),
256
+ .DEPTH ( UserAwMaxTxns ),
257
+ .T ( sel_t ),
258
+ ) i_b_count_fifo (
259
+ .clk_i,
260
+ .rst_ni,
261
+ .flush_i ( 1'b0 ),
262
+ .testmode_i ( 1'b0 ),
263
+ .usage_o ( ),
264
+ .data_i ( aw_sel_out ),
265
+ .valid_i ( mst_req_o.aw_valid & mst_resp_i.aw_ready ),
266
+ .ready_o ( b_progress ),
267
+ .data_o ( b_out ),
268
+ .valid_o ( ), // TODO: Assert true when B comes in (`b_resp_load`)
269
+ .ready_i ( b_end )
270
+ );
271
+ assign b_end = (& b_sel_q) | (b_sel_d == b_out);
272
+ end else begin : gen_no_user_aw_b
273
+ // Simply count payloads as for AW and W
274
+ assign b_end = (& b_sel_q);
275
+ assign b_progress = 1'b1 ;
276
+ end
277
+
208
278
// Output is valid, if it is the last b response for the wide W, we have something
209
279
// in the B FIFO and the B response is valid from the master port.
210
- assign slv_res_o.b_valid = mst_res_i.b_valid & ( & b_sel_q) ;
280
+ assign slv_res_o.b_valid = mst_res_i.b_valid & b_end ;
211
281
212
282
// Assign the b_channel ready output. The master port is ready if something is in the
213
283
// B FIFO. Except, if it is the last one which should do a response on the slave port.
214
- assign mst_req_o.b_ready = ( & b_sel_q) ? slv_req_i.b_ready : 1'b1 ;
284
+ assign mst_req_o.b_ready = b_end ? slv_req_i.b_ready : 1'b1 ;
215
285
// B channel error response retention FF
216
- assign b_sel_d = sel_t ' (b_sel_q + 1'b1 );
217
- assign b_resp_d = ( & b_sel_q) ? axi_pkg :: RESP_OKAY : (b_resp_q | mst_res_i.b.resp);
286
+ assign b_sel_d = sel_t ' (b_end ? '0 : b_sel_q + 1'b1 );
287
+ assign b_resp_d = b_end ? axi_pkg :: RESP_OKAY : (b_resp_q | mst_res_i.b.resp);
218
288
assign b_resp_load = mst_res_i.b_valid & mst_req_o.b_ready;
219
289
`FFLARN (b_sel_q, b_sel_d, b_resp_load, '0 , clk_i, rst_ni)
220
290
`FFLARN (b_resp_q, b_resp_d, b_resp_load, axi_pkg :: RESP_OKAY , clk_i, rst_ni)
@@ -223,6 +293,7 @@ module axi_lite_dw_converter #(
223
293
// Input spill register of the AW channel.
224
294
axi_lite_ar_t ar_chan_spill;
225
295
logic ar_chan_spill_valid, ar_chan_spill_ready;
296
+ logic ar_progress, r_progress,
226
297
227
298
spill_register # (
228
299
.T ( axi_lite_ar_t ),
@@ -238,32 +309,77 @@ module axi_lite_dw_converter #(
238
309
.data_o ( ar_chan_spill )
239
310
);
240
311
241
- sel_t ar_sel_q, ar_sel_d;
312
+ sel_t ar_sel_q, ar_sel_d, ar_sel_out ;
242
313
logic ar_sel_load;
243
314
// AR channel output assignment
244
315
always_comb begin : proc_ar_chan_oup
245
316
mst_req_o.ar = ar_chan_spill;
246
- mst_req_o.ar.addr = out_address (ar_chan_spill.addr, ar_sel_q);
317
+ mst_req_o.ar.addr = out_address (ar_chan_spill.addr, ar_sel_out);
318
+ end
319
+
320
+ assign ar_progress = ar_chan_spill_valid & mst_res_i.ar_ready;
321
+
322
+ if (UserAwSize) begin : gen_user_ar
323
+ // Couple AR and R FIFO
324
+ sel_t ar_sel_end, ar_sel_base;
325
+ assign ar_sel_end = UserAwSize ?
326
+ (sel_t ' (1 ) << ar_chan_spill.user[UserArSizeLsb: + axi_pkg :: SizeWidth]) >> SelOffset : '0 ;
327
+ assign ar_sel_base = ar_chan_spill.ar.addr[SelOffset+ : SelWidth] & sel_t ' (ar_sel_end - 1 );
328
+ assign ar_sel_out = ar_sel_q + ar_sel_base;
329
+ assign mst_req_o.aw_valid = r_progress & ar_chan_spill_valid;
330
+ assign aw_chan_spill_ready = r_progress & mst_res_i.ar_ready &
331
+ ((& ar_sel_q) | (ar_sel_d == ar_sel_end));
332
+ end else begin : gen_no_user_ar
333
+ // AR and R are uncoupled
334
+ assign ar_sel_out = ar_sel_q;
335
+ assign mst_req_o.ar_valid = ar_chan_spill_valid;
336
+ assign ar_chan_spill_ready = mst_res_i.ar_ready & (& ar_sel_q);
247
337
end
248
- // Slave port aw is valid, if there is something in the spill register.
249
- assign mst_req_o.ar_valid = ar_chan_spill_valid;
250
- assign ar_chan_spill_ready = mst_res_i.ar_ready & (& ar_sel_q);
251
338
252
339
assign ar_sel_load = mst_req_o.ar_valid & mst_res_i.ar_ready;
253
- assign ar_sel_d = sel_t ' (ar_sel_q + 1'b1 );
340
+ assign ar_sel_d = sel_t ' (ar_sel_load ? '0 : ar_sel_q + 1'b1 );
254
341
`FFLARN (ar_sel_q, ar_sel_d, ar_sel_load, '0 , clk_i, rst_ni)
255
342
256
343
// Responses have to be aggregated, one FF less, as the last data is feed directly through.
257
344
sel_t r_sel_q, r_sel_d;
258
345
logic r_sel_load;
259
346
axi_lite_mst_r_t [DownsizeFactor- 2 : 0 ] r_chan_mst_q;
260
347
logic [DownsizeFactor- 2 : 0 ] r_chan_mst_load;
348
+ logic r_end;
349
+
350
+ if (UserArSize) begin : gen_user_ar_r
351
+ // When an upstream AR completes, store the expected downstream R count
352
+ sel_t r_out;
353
+ stream_fifo # (
354
+ .FALL_THROUGH ( 1'b0 ),
355
+ .DEPTH ( UserArMaxTxns ),
356
+ .T ( sel_t ),
357
+ ) i_r_count_fifo (
358
+ .clk_i,
359
+ .rst_ni,
360
+ .flush_i ( 1'b0 ),
361
+ .testmode_i ( 1'b0 ),
362
+ .usage_o ( ),
363
+ .data_i ( ar_sel_out ),
364
+ .valid_i ( mst_req_o.ar_valid & mst_resp_i.ar_ready ),
365
+ .ready_o ( r_progress ),
366
+ .data_o ( r_out ),
367
+ .valid_o ( ), // TODO: Assert true when R comes in (`r_sel_load`)
368
+ .ready_i ( r_end )
369
+ );
370
+ assign r_end = (& r_sel_q) | (r_sel_d == r_out);
371
+ end else begin : gen_no_user_ar_r
372
+ // Simply count payloads as for AW and W
373
+ assign r_end = (& r_sel_q);
374
+ assign r_progress = 1'b1 ;
375
+ end
376
+
261
377
for (genvar i = 0 ; unsigned '(i) < (DownsizeFactor- 1 ); i++ ) begin : gen_r_chan_ff
262
378
assign r_chan_mst_load[i] = (sel_t ' (i) == r_sel_q) & mst_res_i.r_valid & mst_req_o.r_ready;
263
379
`FFLARN (r_chan_mst_q[i], mst_res_i.r, r_chan_mst_load[i], axi_lite_mst_r_t'{ default : '0 } , clk_i, rst_ni)
264
380
end
265
381
assign r_sel_load = mst_res_i.r_valid & mst_req_o.r_ready;
266
- assign r_sel_d = sel_t ' (r_sel_q + 1'b1 );
382
+ assign r_sel_d = sel_t ' (r_sel_load ? '0 : r_sel_q + 1'b1 );
267
383
`FFLARN (r_sel_q, r_sel_d, r_sel_load, '0 , clk_i, rst_ni)
268
384
269
385
always_comb begin : proc_r_chan_oup
@@ -273,16 +389,16 @@ module axi_lite_dw_converter #(
273
389
} ;
274
390
// Response is the OR of all responses
275
391
for (int unsigned i = 0 ; i < (DownsizeFactor- 1 ); i++ ) begin
276
- slv_res_o.r.resp = slv_res_o.r.resp | r_chan_mst_q[i].resp;
392
+ slv_res_o.r.resp = slv_res_o.r.resp | (r_sel_q >= sel_t ' (i) ? r_chan_mst_q[i].resp : '0 ) ;
277
393
slv_res_o.r.data[i* AxiMstPortDataWidth+ : AxiMstPortDataWidth] = r_chan_mst_q[i].data;
278
394
end
279
395
// The highest bits of the data can be directly the master port.
280
396
slv_res_o.r.data[(DownsizeFactor- 1 )* AxiMstPortDataWidth+ : AxiMstPortDataWidth] =
281
397
mst_res_i.r.data;
282
398
end
283
399
284
- assign slv_res_o.r_valid = ( & r_sel_q) ? mst_res_i.r_valid : 1'b0 ;
285
- assign mst_req_o.r_ready = ( & r_sel_q) ? slv_req_i.r_ready : 1'b1 ;
400
+ assign slv_res_o.r_valid = r_end ? mst_res_i.r_valid : 1'b0 ;
401
+ assign mst_req_o.r_ready = r_end ? slv_req_i.r_ready : 1'b1 ;
286
402
287
403
end else if (AxiMstPortDataWidth > AxiSlvPortDataWidth) begin : gen_upsizer
288
404
// The upsize factor determines the amount of replication.
0 commit comments