引言
“不如意事常八九,可与语人无二三”,人嘛,没有一帆风顺,一定有这样或那样的异常状况出现。对CPU来说,也是这样,如果一个CPU只能顺序执行,不能处理异常情况,那么这个CPU也就没什么存在的意义了。
本小节,我们就分析一下or1200的异常处理系统。
1,异常
1>异常的概念
这里假定中断是异常的一种类型。
说到异常,大家可能并不陌生,linux里面有异常处理上下文的机制,之前我们在分析MMU模块时有TLB miss异常和page fault异常,在分析cache模块时有对齐异常,还有大家经常听到的中断,这些,都是异常。
2>异常的分类
一般情况下,我们可以把异常分为外部异常(external)和内部异常(internal)两种,其中外部异常,有可以叫中断(interrupt)。
除了这种分类方式外,我们也可以把异常分为可屏蔽异常(maskable)和不可屏蔽异常(unmaskable)两种。如reset,bus error这些异常就是不可屏蔽的。外部中断有的就可以屏蔽。如果某个异常被屏蔽了,那么就意味着这个异常不会打断程序的正常执行顺序。
还有,我们还可以把异常分为同步异常(synchronous)和异步异常(asynchronous)两种。外部中断,复位,这些都是异步的,由指令本身产生的异常一般都是同步的。
对于or1200来说,目前支持的异常类型,异常向量入口,以及异常的产生条件,如下所示。
3>异常的处理过程及精确异常
系统一旦发生异常,CPU会在流水线的WB阶段相应异常,发生异常时,将前一条指令的PC和下一条指令的PC值进行保存,以及CPU运行状态的现场保存,然后将分辨是哪种异常,并把对应的异常处理的入口赋给PC,开始异常处理程序,在执行完异常处理程序之后,进行异常发生前现场的恢复,CPU恢复执行。
如果CPU能保证发生异常的那条指令前的所有指令都已执行完毕,那条指令后面的所有指令都没有执行,那么,这个CPU就称作是精确异常CPU(precise except)。
从上面的异常处理过程,我们可以发现,异常处理的开始和结束时需要保存和回复CPU现场,如果异常比较频繁,那么现场保存和恢复的开销就比较大,为了减小异常的现场保存与恢复开销,一般会设计一定量的shadow register,如果shadow register越多,支持的异常嵌套(nested exception)层数就越多。
关于or1200的异常模型的一些介绍,如异常的优先级,异常发生时的延迟槽的处理,以及异常发生时的超级用户模式切换,快速上下文切换等问题,请参考架构手册的except mode章节,这里不再赘述。
2,整体结构
在了解了异常的一些内容之后,我们再来分析or1200中异常处理系统。首先是except系统的整体结构,如下所示:
需要说明的有以下几点:
a,or1200的外部中断源有pic模块统一处理,最多支持的外部中断线的数量是32条。关于or1200的外部中断体统的分析,使用和验证,我们之前已经介绍过了,如有疑问,请参考:http://blog.csdn.net/rill_zhen/article/details/8894856
b,or1200的异常处理核心是在cpu内部的except模块,对应的rtl文件是or1200_except.v,这个文件,我们会在下面进行详细分析。
c,tt模块(tick timer),这个模块的作用是为操作系统或其它应用程序提供精确的定时服务。就像我们在学习单片机时,使用的定时器一样。
3,pic模块
1>功能介绍
pic模块(programable interrupt controller),负责收集来自CPU外部的所有异常信号,比如来自ethernet的中断信号,来自SD card controller的中断信号,等等等等。
一但有中断产生,pic模块会发送一个intr信号给CPU内部的except模块,except会经过进一步处理,并将外部中断的异常处理向量入口地址(0x800)赋给PC,异常处理程序会使用mfspr指令,通过sprs模块将32-bit的中断信号全部读出到reg_file(通用寄存器),然后确定是哪条中线线发生了中断,然后执行这条中断线注册的中断处理函数。当然,这个过程中包括现场保存和恢复工作。
总之,pic模块的作用比较简单,就是一旦有外部中断产生,将所有32-bit的中断值进行保存,并产生一个intr信号告诉except模块,然后等待sprs的读取。
此外,pic模块还可以通过sprs模块来设置哪些中断信号被屏蔽。
2>中断信号的处理
pic模块内部有两个寄存器,picmr(pic mask register)和picsr(pic status register)。
picmr是屏蔽寄存器,如果这个寄存器的某一位被sprs模块设置为1,那么就表示这一位对应的中断将被忽略。
picsr是状态寄存器,一旦有哪根中断线产生中断,那么对应的位将会置1。
这里需要说明的是,
a,目前or1200外部中断线是20个,or1200_define中定义如下:
// Define number of interrupt inputs (2-31)
`define OR1200_PIC_INTS 20
b,其中第0,1两个中断是非屏蔽中断。
c,pic模块还有另外一个功能就是,一旦发生外部中断,会马上给pm(power management)发送一个唤醒信号,pic_wakeup。
d,当然,pic模块会相应sprs模块对picmr和picsr两个寄存器的读写操作。
pic模块对应的rtl文件是or1200_pic.v,其核心代码如下:
// synopsys translate_off
`include "timescale.v"
// synopsys translate_on
`include "or1200_defines.v"
module or1200_pic(
// RISC Internal Interface
clk, rst, spr_cs, spr_write, spr_addr, spr_dat_i, spr_dat_o,
pic_wakeup, intr,
// PIC Interface
pic_int
);
//
// PIC Interface
//
input [`OR1200_PIC_INTS-1:0] pic_int;// Interrupt inputs
//
// PIC Mask Register bits (or no register)
//
reg [`OR1200_PIC_INTS-1:2] picmr; // PICMR bits
//
// PIC Status Register bits (or no register)
//
reg [`OR1200_PIC_INTS-1:0] picsr; // PICSR bits
//
// Internal wires & regs
//
wire picmr_sel; // PICMR select
wire picsr_sel; // PICSR select
wire [`OR1200_PIC_INTS-1:0] um_ints;// Unmasked interrupts
reg [31:0] spr_dat_o; // SPR data out
//
// PIC registers address decoder
//
assign picmr_sel = (spr_cs && (spr_addr[`OR1200_PICOFS_BITS] == `OR1200_PIC_OFS_PICMR)) ? 1'b1 : 1'b0;
assign picsr_sel = (spr_cs && (spr_addr[`OR1200_PICOFS_BITS] == `OR1200_PIC_OFS_PICSR)) ? 1'b1 : 1'b0;
//
// Write to PICMR
//
always @(posedge clk or `OR1200_RST_EVENT rst)
if (rst == `OR1200_RST_VALUE)
picmr <= {1'b1, {`OR1200_PIC_INTS-3{1'b0}}};
else if (picmr_sel && spr_write) begin
picmr <= spr_dat_i[`OR1200_PIC_INTS-1:2];
end
//
// Write to PICSR, both CPU and external ints
//
always @(posedge clk or `OR1200_RST_EVENT rst)
if (rst == `OR1200_RST_VALUE)
picsr <= {`OR1200_PIC_INTS{1'b0}};
else if (picsr_sel && spr_write) begin
picsr <= spr_dat_i[`OR1200_PIC_INTS-1:0] | um_ints;
end else
picsr <= picsr | um_ints;
//
// Unmasked interrupts
//
assign um_ints = pic_int & {picmr, 2'b11};
//
// Generate intr
//
assign intr = |um_ints;
//
// Assert pic_wakeup when intr is asserted
//
assign pic_wakeup = intr;
//
// Read PIC registers
//
always @(spr_addr or picmr or picsr)
case (spr_addr[`OR1200_PICOFS_BITS]) // synopsys parallel_case
`OR1200_PIC_OFS_PICMR: begin
spr_dat_o[`OR1200_PIC_INTS-1:0] = {picmr, 2'b11};
spr_dat_o[31:`OR1200_PIC_INTS] = {32-`OR1200_PIC_INTS{1'b0}};
end
default: begin
spr_dat_o[`OR1200_PIC_INTS-1:0] = picsr;
spr_dat_o[31:`OR1200_PIC_INTS] = {32-`OR1200_PIC_INTS{1'b0}};
end
endcase
endmodule
从这段代码中,我们除了了解pic模块的功能外,我们还需要知道两点:
a,在or1200的整个工程中,几乎所有的verilog文件的开头都会有下面三行代码,为什么呢?
// synopsys translate_off
`include "timescale.v"
// synopsys translate_on
这三行代码,其中两行是注释,这个注释可不是一般的注释,这两行注释是有其特殊含义的。由于verilog HDL没有像VHDL中的属性语句,所以为了告诉综合器一些信息,就需要通过注释来实现。这两行注释就是给synopsys的工具识别的,表示在综合时忽略中间的那行include语句。
但是给综合器使用的特殊注释语句,在用modelsim等仿真工具时是无效的。
b,同样是特殊注释语句,我们也经常看到像下面的一条注释,那么,这条注释又起到什么作用呢?
case (spr_addr[`OR1200_PICOFS_BITS]) // synopsys parallel_case
这条注释也是告诉综合器,在综合时这个case块使用parallel case。这会对综合产生什么影响呢?
这就需要了解full case和parallel case的区别。简单来说,非full case块会综合成优先级逻辑电路,full case和parallel case会综合成多路选择器电路。
如果代码中写的是非full case语句,但是想让综合器综合成多路选择器,可能如果有相同的case入口,就会产生错误的电路。为了避免这种情况出现,我们在写case块时,要保证将所有的可能性全部列出来,并以default结束。
4,except模块
1>整体功能
except模块,对应的rtl文件是or1200_except.v,此模块在cpu内部,是CPU异常处理的核心模块。这个模块会收集所有的异常信号,分析异常类型,并根据事先确定的异常处理操作。例如,设置EXPC,也就是将异常向量入口地址赋给PC;暂停或刷新流水线等。
except模块一旦发现有异常产生,就会发送except_start信号,告诉其他模块,要进行异常处理了,之后发送except_started信号,告诉其他模块正在进行异常处理,一旦异常处理完成,会发送except_stop信号。当然except模块也会将一部分自己不能处理的异常信号传送出去,这个信号就是except_trig。
2>模块分析
了解了except的整体功能,我们再来看其rtl源码就简单一写了。
首先是异常的总体收集,主要代码如下:
//
// Order defines exception detection priority
//
assign except_trig = {
ex_exceptflags[1] & ~du_dsr[`OR1200_DU_DSR_IME],
ex_exceptflags[0] & ~du_dsr[`OR1200_DU_DSR_IPFE],
ex_exceptflags[2] & ~du_dsr[`OR1200_DU_DSR_BUSEE],
sig_illegal & ~du_dsr[`OR1200_DU_DSR_IIE],
sig_align & ~du_dsr[`OR1200_DU_DSR_AE],
sig_dtlbmiss & ~du_dsr[`OR1200_DU_DSR_DME],
sig_trap & ~du_dsr[`OR1200_DU_DSR_TE],
sig_syscall & ~du_dsr[`OR1200_DU_DSR_SCE] & ~ex_freeze,
sig_dmmufault & ~du_dsr[`OR1200_DU_DSR_DPFE],
sig_dbuserr & ~du_dsr[`OR1200_DU_DSR_BUSEE],
range_pending & ~du_dsr[`OR1200_DU_DSR_RE],
fp_pending & ~du_dsr[`OR1200_DU_DSR_FPE],
int_pending & ~du_dsr[`OR1200_DU_DSR_IE],
tick_pending & ~du_dsr[`OR1200_DU_DSR_TTE]
};
这个是整体的收集工作,具体某个部分的异常信号,有其单独的收集代码,如下所示:
assign int_pending = sig_int & (sr[`OR1200_SR_IEE] |
(sr_we & to_sr[`OR1200_SR_IEE]))
& id_pc_val & delayed_iee[2] & ~ex_freeze & ~ex_branch_taken
& ~ex_dslot & ~(sr_we & ~to_sr[`OR1200_SR_IEE]);
assign tick_pending = sig_tick & (sr[`OR1200_SR_TEE] |
(sr_we & to_sr[`OR1200_SR_TEE])) & id_pc_val
& delayed_tee[2] & ~ex_freeze & ~ex_branch_taken
& ~ex_dslot & ~(sr_we & ~to_sr[`OR1200_SR_TEE]);
assign fp_pending = sig_fp & fpcsr_fpee & ~ex_freeze & ~ex_branch_taken
& ~ex_dslot;
assign range_pending = sig_range & sr[`OR1200_SR_OVE] & ~ex_freeze &
~ex_branch_taken & ~ex_dslot;
一旦出现异常,就需要设计开始和已经开始,以及在异常处理结束时设置结束信号,代码如下:
assign except_started = extend_flush & except_start;
assign except_start = (except_type != `OR1200_EXCEPT_NONE) & extend_flush;
assign except_stop = {
tick_pending & du_dsr[`OR1200_DU_DSR_TTE],
int_pending & du_dsr[`OR1200_DU_DSR_IE],
ex_exceptflags[1] & du_dsr[`OR1200_DU_DSR_IME],
ex_exceptflags[0] & du_dsr[`OR1200_DU_DSR_IPFE],
ex_exceptflags[2] & du_dsr[`OR1200_DU_DSR_BUSEE],
sig_illegal & du_dsr[`OR1200_DU_DSR_IIE],
sig_align & du_dsr[`OR1200_DU_DSR_AE],
sig_dtlbmiss & du_dsr[`OR1200_DU_DSR_DME],
sig_dmmufault & du_dsr[`OR1200_DU_DSR_DPFE],
sig_dbuserr & du_dsr[`OR1200_DU_DSR_BUSEE],
range_pending & du_dsr[`OR1200_DU_DSR_RE],
sig_trap & du_dsr[`OR1200_DU_DSR_TE],
fp_pending & du_dsr[`OR1200_DU_DSR_FPE],
sig_syscall & du_dsr[`OR1200_DU_DSR_SCE] & ~ex_freeze
};
在开始执行异常处理程序之前,要进行现场保护,代码如下:
always @(posedge clk or `OR1200_RST_EVENT rst) begin
if (rst == `OR1200_RST_VALUE) begin
ex_freeze_prev <= 1'b0 ;
sr_ted_prev <= 1'b0 ;
dsr_te_prev <= 1'b0 ;
dmr1_st_prev <= 1'b0 ;
dmr1_bt_prev <= 1'b0 ;
end
else begin
ex_freeze_prev <= ex_freeze ;
if (!ex_freeze_prev || ex_void) begin
sr_ted_prev <= sr [`OR1200_SR_TED ] ;
dsr_te_prev <= du_dsr [`OR1200_DU_DSR_TE ] ;
dmr1_st_prev <= du_dmr1[`OR1200_DU_DMR1_ST] ;
dmr1_bt_prev <= du_dmr1[`OR1200_DU_DMR1_BT] ;
end
end
end
在保存完现场之后,就可以根据不同的异常类型,确定其事先约定好的异常处理流程。
不同的异常类型,对应不同的异常操作,无论是哪种异常都需要将异常向量地址赋给PC,代码如下:
这个过程分为两小步,首先是确定异常类型,并将对应的地址赋给相应的信号:
//
// Exception FSM that sequences execution of exception handler
//
// except_type signals which exception handler we start fetching in:
// 1. Asserted in next clock cycle after exception is recognized
//
always @(posedge clk or `OR1200_RST_EVENT rst) begin
if (rst == `OR1200_RST_VALUE) begin
state <= `OR1200_EXCEPTFSM_IDLE;
except_type <= `OR1200_EXCEPT_NONE;
extend_flush <= 1'b0;
epcr <= 32'b0;
eear <= 32'b0;
esr <= {2'h1, {`OR1200_SR_WIDTH-3{1'b0}}, 1'b1};
extend_flush_last <= 1'b0;
end
else begin
`ifdef OR1200_CASE_DEFAULT
case (state) // synopsys parallel_case
`else
case (state) // synopsys full_case parallel_case
`endif
`OR1200_EXCEPTFSM_IDLE:
if (except_flushpipe) begin
state <= `OR1200_EXCEPTFSM_FLU1;
extend_flush <= 1'b1;
esr <= sr_we ? to_sr : sr;
casez (except_trig)
`ifdef OR1200_EXCEPT_ITLBMISS
14'b1?_????_????_????: begin
except_type <= `OR1200_EXCEPT_ITLBMISS;
eear <= ex_dslot ?
ex_pc : ex_pc;
epcr <= ex_dslot ?
wb_pc : ex_pc;
end
`endif
`ifdef OR1200_EXCEPT_IPF
14'b01_????_????_????: begin
except_type <= `OR1200_EXCEPT_IPF;
eear <= ex_dslot ?
ex_pc : delayed1_ex_dslot ?
id_pc : delayed2_ex_dslot ?
id_pc : id_pc;
epcr <= ex_dslot ?
wb_pc : delayed1_ex_dslot ?
id_pc : delayed2_ex_dslot ?
id_pc : id_pc;
end
`endif
`ifdef OR1200_EXCEPT_BUSERR
14'b00_1???_????_????: begin // Insn. Bus Error
except_type <= `OR1200_EXCEPT_BUSERR;
eear <= ex_dslot ?
wb_pc : ex_pc;
epcr <= ex_dslot ?
wb_pc : ex_pc;
end
`endif
`ifdef OR1200_EXCEPT_ILLEGAL
14'b00_01??_????_????: begin
except_type <= `OR1200_EXCEPT_ILLEGAL;
eear <= ex_pc;
epcr <= ex_dslot ?
wb_pc : ex_pc;
end
`endif
`ifdef OR1200_EXCEPT_ALIGN
14'b00_001?_????_????: begin
except_type <= `OR1200_EXCEPT_ALIGN;
eear <= lsu_addr;
epcr <= ex_dslot ?
wb_pc : ex_pc;
end
`endif
`ifdef OR1200_EXCEPT_DTLBMISS
14'b00_0001_????_????: begin
except_type <= `OR1200_EXCEPT_DTLBMISS;
eear <= lsu_addr;
epcr <= ex_dslot ?
wb_pc : delayed1_ex_dslot ?
dl_pc : ex_pc;
end
`endif
`ifdef OR1200_EXCEPT_TRAP
14'b00_0000_1???_????: begin
except_type <= `OR1200_EXCEPT_TRAP;
epcr <= ex_dslot ?
wb_pc : delayed1_ex_dslot ?
id_pc : ex_pc;
end
`endif
`ifdef OR1200_EXCEPT_SYSCALL
14'b00_0000_01??_????: begin
except_type <= `OR1200_EXCEPT_SYSCALL;
epcr <= ex_dslot ?
wb_pc : delayed1_ex_dslot ?
id_pc : delayed2_ex_dslot ?
id_pc : id_pc;
end
`endif
`ifdef OR1200_EXCEPT_DPF
14'b00_0000_001?_????: begin
except_type <= `OR1200_EXCEPT_DPF;
eear <= lsu_addr;
epcr <= ex_dslot ?
wb_pc : delayed1_ex_dslot ?
dl_pc : ex_pc;
end
`endif
`ifdef OR1200_EXCEPT_BUSERR
14'b00_0000_0001_????: begin // Data Bus Error
except_type <= `OR1200_EXCEPT_BUSERR;
eear <= lsu_addr;
epcr <= ex_dslot ?
wb_pc : delayed1_ex_dslot ?
dl_pc : ex_pc;
end
`endif
`ifdef OR1200_EXCEPT_RANGE
14'b00_0000_0000_1???: begin
except_type <= `OR1200_EXCEPT_RANGE;
epcr <= ex_dslot ?
wb_pc : delayed1_ex_dslot ?
id_pc : delayed2_ex_dslot ?
id_pc : id_pc;
end
`endif
`ifdef OR1200_EXCEPT_FLOAT
14'b00_0000_0000_01??: begin
except_type <= `OR1200_EXCEPT_FLOAT;
epcr <= id_pc;
end
`endif
`ifdef OR1200_EXCEPT_INT
14'b00_0000_0000_001?: begin
except_type <= `OR1200_EXCEPT_INT;
epcr <= id_pc;
end
`endif
`ifdef OR1200_EXCEPT_TICK
14'b00_0000_0000_0001: begin
except_type <= `OR1200_EXCEPT_TICK;
epcr <= id_pc;
end
`endif
default:
except_type <= `OR1200_EXCEPT_NONE;
endcase
end
else if (pc_we) begin
state <= `OR1200_EXCEPTFSM_FLU1;
extend_flush <= 1'b1;
end
else begin
if (epcr_we)
epcr <= datain;
if (eear_we)
eear <= datain;
if (esr_we)
esr <= {datain[`OR1200_SR_WIDTH-1], 1'b1, datain[`OR1200_SR_WIDTH-3:0]};
end
`OR1200_EXCEPTFSM_FLU1:
if (icpu_ack_i | icpu_err_i | genpc_freeze)
state <= `OR1200_EXCEPTFSM_FLU2;
`OR1200_EXCEPTFSM_FLU2:
`ifdef OR1200_EXCEPT_TRAP
if (except_type == `OR1200_EXCEPT_TRAP) begin
state <= `OR1200_EXCEPTFSM_IDLE;
extend_flush <= 1'b0;
extend_flush_last <= 1'b0;
except_type <= `OR1200_EXCEPT_NONE;
end
else
`endif
state <= `OR1200_EXCEPTFSM_FLU3;
`OR1200_EXCEPTFSM_FLU3:
begin
state <= `OR1200_EXCEPTFSM_FLU4;
end
`OR1200_EXCEPTFSM_FLU4: begin
state <= `OR1200_EXCEPTFSM_FLU5;
extend_flush <= 1'b0;
extend_flush_last <= 1'b0; // damjan
end
`ifdef OR1200_CASE_DEFAULT
default: begin
`else
`OR1200_EXCEPTFSM_FLU5: begin
`endif
if (!if_stall && !id_freeze) begin
state <= `OR1200_EXCEPTFSM_IDLE;
except_type <= `OR1200_EXCEPT_NONE;
extend_flush_last <= 1'b0;
end
end
endcase
end
end
由于异常有很多类型,但流水线只有一条,所以第二小步就是将不同类型的地址信号赋给最终要执行的PC,代码如下:
//
// PC and Exception flags pipelines
//
always @(posedge clk or `OR1200_RST_EVENT rst) begin
if (rst == `OR1200_RST_VALUE) begin
wb_pc <= 32'd0;
dl_pc <= 32'd0;
end
else if (!wb_freeze) begin
wb_pc <= ex_pc;
dl_pc <= wb_pc;
end
end
在异常发生后,我们还要更新位于sprs模块里面的SR寄存器,关于SR寄存器的,我们在分析SPRS模块时已经介绍过了,如有疑问请参考sprs的内容。
except模块是如何设置SR寄存器的呢?代码如下:
//
// delayed_iee
//
// SR[IEE] should not enable interrupts right away
// when it is restored with l.rfe. Instead delayed_iee
// together with SR[IEE] enables interrupts once
// pipeline is again ready.
//
always @(`OR1200_RST_EVENT rst or posedge clk)
if (rst == `OR1200_RST_VALUE)
delayed_iee <= 3'b000;
else if (!sr[`OR1200_SR_IEE])
delayed_iee <= 3'b000;
else
delayed_iee <= {delayed_iee[1:0], 1'b1};
//
// delayed_tee
//
// SR[TEE] should not enable tick exceptions right away
// when it is restored with l.rfe. Instead delayed_tee
// together with SR[TEE] enables tick exceptions once
// pipeline is again ready.
//
always @(`OR1200_RST_EVENT rst or posedge clk)
if (rst == `OR1200_RST_VALUE)
delayed_tee <= 3'b000;
else if (!sr[`OR1200_SR_TEE])
delayed_tee <= 3'b000;
else
delayed_tee <= {delayed_tee[1:0], 1'b1};
wire dsr_te = ex_freeze_prev ? dsr_te_prev : du_dsr[`OR1200_DU_DSR_TE];
wire sr_ted = ex_freeze_prev ? sr_ted_prev : sr[`OR1200_SR_TED];
在异常处理真正开始之前,except模块除了要保护现场之外,有些正在执行的指令需要做一些特殊处理,比如load/store指令,mtspr/mfspr指令等,代码如下:
// Abort write into RF by load & other instructions
assign abort_ex = sig_dbuserr | sig_dmmufault | sig_dtlbmiss | sig_align |
sig_illegal | ((du_hwbkpt | trace_trap) & ex_pc_val
& !sr_ted & !dsr_te);
// abort spr read/writes
assign abort_mvspr = sig_illegal | ((du_hwbkpt | trace_trap) & ex_pc_val
& !sr_ted & !dsr_te) ;
此外,由于异常的处理是在流水线的最后一个阶段,WB阶段才被处理,所以就需要刷新流水线,这样,就需要处理在之前已经进入流水线的指令,例如,要判断异常处理入口的指令是否在延迟槽内,如何暂停和刷新流水线等,代码如下:
//
// PC and Exception flags pipelines
//
always @(posedge clk or `OR1200_RST_EVENT rst) begin
if (rst == `OR1200_RST_VALUE) begin
id_pc <= 32'd0;
id_pc_val <= 1'b0 ;
id_exceptflags <= 3'b000;
end
else if (id_flushpipe) begin
id_pc_val <= 1'b0 ;
id_exceptflags <= 3'b000;
end
else if (!id_freeze) begin
id_pc <= if_pc;
id_pc_val <= 1'b1 ;
id_exceptflags <= { sig_ibuserr, sig_itlbmiss, sig_immufault };
end
end
//
// PC and Exception flags pipelines
//
always @(posedge clk or `OR1200_RST_EVENT rst) begin
if (rst == `OR1200_RST_VALUE) begin
ex_dslot <= 1'b0;
ex_pc <= 32'd0;
ex_pc_val <= 1'b0 ;
ex_exceptflags <= 3'b000;
delayed1_ex_dslot <= 1'b0;
delayed2_ex_dslot <= 1'b0;
end
else if (ex_flushpipe) begin
ex_dslot <= 1'b0;
ex_pc_val <= 1'b0 ;
ex_exceptflags <= 3'b000;
delayed1_ex_dslot <= 1'b0;
delayed2_ex_dslot <= 1'b0;
end
else if (!ex_freeze & id_freeze) begin
ex_dslot <= 1'b0;
ex_pc <= id_pc;
ex_pc_val <= id_pc_val ;
ex_exceptflags <= 3'b000;
delayed1_ex_dslot <= ex_dslot;
delayed2_ex_dslot <= delayed1_ex_dslot;
end
else if (!ex_freeze) begin
ex_dslot <= ex_branch_taken;
ex_pc <= id_pc;
ex_pc_val <= id_pc_val ;
ex_exceptflags <= id_exceptflags;
delayed1_ex_dslot <= ex_dslot;
delayed2_ex_dslot <= delayed1_ex_dslot;
end
end
其实,except模块是一个整体,我们之所以拆开了分析是想能分析的更透彻一些,但是拆开来分析的话,也会造成缺乏整体感,尤其是对于像异常处理系统,这样的控制模块。
CPU的控制通路和数据通路的最大区别就是,数据通路的各个模块之间的联系比较少,模块接口简单。控制通路的模块,端口较多,并且信号控制复杂而庞大,这就增加了对控制模块的分析难度。所以为了更清楚了解析or1200的控制通路,我采用了先整体,后分离的方式。先介绍模块的整体功能,将模块的功能进行分割,然后逐个分析每个小的功能部件,最终分析每个部件的rtl代码。
5,小结
except,可以说是or1200 cpu内部控制逻辑最复杂的模块之一,本小节对这个模块进行了初步的分析,以后还会有更准确,详细的持续分析,我已将后续的分析内容增加到这里。
enjoy!