OpenRisc-49-or1200的except模块分析

引言

“不如意事常八九,可与语人无二三”,人嘛,没有一帆风顺,一定有这样或那样的异常状况出现。对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!


评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值