forked from ggeeyyi/loongson-cpu
-
Notifications
You must be signed in to change notification settings - Fork 0
/
IF_stage.v
193 lines (162 loc) · 6.38 KB
/
IF_stage.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
`include "mycpu.h"
module IF_stage (
/*
IF_stage模块
clk: 时钟
resetn: 复位信号
ID_allowin: 允许ID_stage接收数据
br_bus: 分支总线
IF_ID_valid: IF_ID是否有效
IF_ID_bus: IF_ID总线
WB_EXC_signal: 例外信号
WB_ERTN_signal: 例外返回信号
CSR_2_IF_pc: csr到IF_stage的pc
inst_sram_req: 指令存储器请求
inst_sram_wr: 指令存储器写使能
inst_sram_size: 传输数据大小
inst_sram_wstrb: 请求写数据的掩码
inst_sram_addr: 取指时输入指令存储器的地址
inst_sram_wdata: 指令存储器写数据
inst_sram_addr_ok: 指令存储器地址有效
inst_sram_data_ok: 指令存储器数据有效
inst_sram_rdata: 指令存储器读数据
*/
input wire clk,
input wire resetn,
// ID to IF interface
input wire ID_allowin,
input wire [ 33:0] br_bus,
// IF to ID interface
output wire IF_ID_valid,
output wire [`IF_ID_LEN -1:0] IF_ID_bus,
input wire WB_EXC_signal,
input wire WB_ERTN_signal,
input wire [31:0] CSR_2_IF_pc,
// inst sram interface
output wire inst_sram_req,
output wire inst_sram_wr,
output wire [ 1:0] inst_sram_size,
output wire [ 3:0] inst_sram_wstrb,
output wire [31:0] inst_sram_addr,
output wire [31:0] inst_sram_wdata,
input wire inst_sram_addr_ok,
input wire inst_sram_data_ok,
input wire [31:0] inst_sram_rdata,
input wire [3:0] axi_arid,
input wire ID_exc_syscall
);
reg IF_valid;
wire IF_ready_go;
wire IF_allowin;
wire to_IF_valid;
wire br_stall;
wire [31:0] seq_pc;
wire [31:0] nextpc;
wire br_taken;
wire [31:0] br_target;
reg pre_IF_block;
assign {br_stall, br_taken, br_target} = br_bus;
wire [31:0] IF_inst;
reg [31:0] IF_pc;
wire IF_exc_ADEF;
reg inst_sram_addr_suc;
wire pre_IF_ready_go;
reg pre_IF_EXC_signal;
reg inst_buffer_valid;
reg [31:0] inst_buffer;
reg PC_buf_valid;
reg [31:0] PC_buffer;
wire IF_cancel;
reg inst_discard;
//------------------------------pre-IF signal----------------------------------------------
assign pre_IF_ready_go = inst_sram_req & inst_sram_addr_ok;
assign to_IF_valid = pre_IF_ready_go & ~IF_cancel & ~pre_IF_block;
assign seq_pc = IF_pc + 3'h4;
assign nextpc = ID_exc_syscall? 32'h1c008000:
(WB_EXC_signal | WB_ERTN_signal)? CSR_2_IF_pc : PC_buf_valid? PC_buffer :br_taken? br_target: seq_pc;
always @(posedge clk) begin
if (~resetn) pre_IF_block <= 1'b0;
else if (IF_cancel & ~pre_IF_block & ~axi_arid[0] & ~inst_sram_data_ok) pre_IF_block <= 1'b1;
else if (inst_sram_data_ok) pre_IF_block <= 1'b0;
end
//------------------------------state control signal---------------------------------------
assign IF_ready_go = (inst_sram_data_ok|inst_buffer_valid) & ~ inst_discard; //存在有效返回值
assign IF_allowin = ~IF_valid | IF_ready_go & ID_allowin;
assign IF_ID_valid = IF_valid & IF_ready_go;
always @(posedge clk) begin
if (~resetn) IF_valid <= 1'b0;
else if (IF_allowin) //此时允许进入新的指令
IF_valid <= to_IF_valid;
else if (IF_cancel) IF_valid <= 1'b0;
end
//------------------------------IF TO ID state interface---------------------------------------
//IF_pc存前一条指令的pc值
always @(posedge clk) begin
if (~resetn) IF_pc <= 32'h1BFF_FFFC;
else if (IF_allowin & to_IF_valid) IF_pc <= nextpc;
end
// 设置寄存器,暂存指令,并用valid信号表示其内指令是否有效
always @(posedge clk) begin
if (~resetn) begin
inst_buffer_valid <= 1'b0;
inst_buffer <= 32'h0;
end else if (IF_ID_valid & !ID_allowin) begin //
inst_buffer_valid <= 1'b1;
inst_buffer <= inst_sram_rdata;
end else if (IF_cancel && inst_buffer_valid && !IF_allowin && IF_ready_go) begin
inst_buffer_valid <= 1'b0;
end else begin
inst_buffer_valid <= 1'b0;
inst_buffer <= 32'h0;
end
end
assign IF_inst = inst_buffer_valid ? inst_buffer : inst_sram_rdata;
assign IF_ID_bus = {IF_exc_ADEF, IF_inst, IF_pc};
always @(posedge clk) begin
if(~resetn)
inst_sram_addr_suc <= 1'b0;
else if(pre_IF_ready_go)
inst_sram_addr_suc <= 1'b1;
else if(inst_sram_data_ok)
inst_sram_addr_suc <= 1'b0;
end
//------------------------------inst sram interface---------------------------------------
assign inst_sram_req = IF_allowin & resetn & ~br_stall& ~pre_IF_block & ~inst_sram_addr_suc; // IF_allow置0时,不请求数据 ***
assign inst_sram_wr = |inst_sram_wstrb; //置0,IF_stage不写入指令存储器
assign inst_sram_wstrb = 4'b0; // 置0,IF_stage不写入指令存储器
assign inst_sram_addr = nextpc;
assign inst_sram_wdata = 32'b0; // 置0,IF_stage不写入指令存储器
assign inst_sram_size = 3'b0;
//------------------------------cancel signal----------------------------------------------
assign IF_cancel = WB_EXC_signal | WB_ERTN_signal | br_taken;
//丢弃指令部分
always @(posedge clk) begin
if (~resetn) inst_discard <= 1'b0;
else if (IF_cancel & ~IF_allowin & ~IF_ready_go | IF_cancel & inst_sram_req) //allowin=0且ready_go=0时
inst_discard <= 1'b1;
else if (inst_discard & inst_sram_data_ok) //需要抹去一条指令
inst_discard <= 1'b0;
end
//------------------------------PC buffer----------------------------------------------
// 使用缓冲寄存器存储异常返回地址或者分支跳转地址,以避免等待addr_ok信号的过程中pc值被覆盖
always @(posedge clk) begin
if (~resetn) begin
PC_buf_valid <= 1'b0;
PC_buffer <= 32'h0;
end else if (WB_EXC_signal) begin
PC_buffer <= CSR_2_IF_pc;
PC_buf_valid <= 1'b1;
end else if (WB_ERTN_signal) begin
PC_buffer <= CSR_2_IF_pc;
PC_buf_valid <= 1'b1;
end else if (br_taken) begin
PC_buffer <= br_target;
PC_buf_valid <= 1'b1;
end else if (pre_IF_ready_go) begin
PC_buf_valid <= 1'b0;
PC_buffer <= 32'h0;
end
end
//---------------------------------exception signal-----------------------------------------
assign IF_exc_ADEF = (|IF_pc[1:0]) & IF_valid;
endmodule