-
Notifications
You must be signed in to change notification settings - Fork 385
/
Copy pathread_ahead_buf.sv
executable file
·143 lines (121 loc) · 3.6 KB
/
read_ahead_buf.sv
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
//------------------------------------------------------------------------------
// fwft_read_ahead_buf.sv
// published as part of https://github.com/pConst/basic_verilog
// Konstantin Pavlov, [email protected]
//------------------------------------------------------------------------------
// INFO ------------------------------------------------------------------------
// Read ahead buffer for FWFT fifo
//
// The buffer substitutes fifo read port and performs fifo data update at the
// same clock cycle as r_req, combinationally, for a one cycle earlier than
// it is expected from standard FWFT fifo
//
// Featires:
// - effectively increases fifo depth by one word
// - adds one cycle lateny for empty flag deassertion
// - does not touch fifo write port and full flag operation logic
// - hides all combinatorial tinkering inside
// - allows controlling and analizing fifo read signals from
// a single always_ff block, like this:
//
// always_ff @(posedge clk) begin
//
// // read control logic
// if( ~empty ) begin // masking rd_req in always_ff
// r_req <= 1'b1;
// end else begin
// r_req <= 1'b0;
// end
//
// // getting input data
// if( r_req ) begin
// new_data[] <= r_data[]; // getting data in always_ff
// end
//
// end
// end
//
/* --- INSTANTIATION TEMPLATE BEGIN ---
read_ahead_buf #(
.DATA_W( 32 )
) RB1 (
.clk( ),
.anrst( ),
// input fifo interface
.fifo_r_req( ),
.fifo_r_data( ),
.fifo_empty( ),
// output fifo interface
.r_req( ),
.r_data( ),
.empty( )
);
--- INSTANTIATION TEMPLATE END ---*/
// synopsys translate_off
`define SIMULATION yes
// synopsys translate_on
module read_ahead_buf #( parameter
DATA_W = 32
)(
input clk, // clock
input anrst, // inverse reset
// input fifo interface
output fifo_r_req,
input [DATA_W-1:0] fifo_r_data,
input fifo_empty,
// output fifo interface
input r_req,
output logic [DATA_W-1:0] r_data,
output logic empty
);
logic fifo_empty_fall;
edge_detect fifo_empty_ed (
.clk( clk ),
.anrst( anrst ),
.in( fifo_empty ),
.rising( ),
.falling( fifo_empty_fall ),
.both( )
);
logic fantom_read;
logic normal_read;
logic buf_empty = 1'b1;
always_ff @(posedge clk or negedge anrst) begin
if( ~anrst ) begin
buf_empty = 1'b1;
end else begin
if( fantom_read ) begin
buf_empty <= 1'b0;
end else if( fifo_empty && r_req ) begin
buf_empty = 1'b1;
end
end
end
assign fantom_read = fifo_empty_fall && buf_empty;
assign normal_read = r_req && ~fifo_empty;
assign empty = buf_empty || // empty falls only after fantom read
(r_req && fifo_empty); // early empty assertion
assign fifo_r_req = anrst &&
(fantom_read || normal_read);
// prepare combinational signal for soft_latch
logic latch_req;
logic [DATA_W-1:0] r_data_latch;
always_comb begin
latch_req = ( fantom_read || normal_read ) ||
( fifo_empty && r_req ); // buffer depletion
if( latch_req ) begin
r_data_latch[DATA_W-1:0] = fifo_r_data[DATA_W-1:0];
end else begin
r_data_latch[DATA_W-1:0] = '0;
end
end
soft_latch #(
.WIDTH( DATA_W )
) r_data_latch_b (
.clk( clk ),
.anrst( anrst ),
.latch( latch_req ),
.in( r_data_latch[DATA_W-1:0] ),
.out( r_data[DATA_W-1:0] )
);
endmodule