@ -60,65 +60,90 @@ entity execute1 is
end entity execute1;
end entity execute1;
architecture behaviour of execute1 is
architecture behaviour of execute1 is
type reg_type is record
type side_effect_type is record
terminate : std_ulogic;
icache_inval : std_ulogic;
write_msr : std_ulogic;
write_xerlow : std_ulogic;
write_dec : std_ulogic;
write_cfar : std_ulogic;
write_loga : std_ulogic;
inc_loga : std_ulogic;
write_pmuspr : std_ulogic;
end record;
constant side_effect_init : side_effect_type := (others => '0');
type actions_type is record
e : Execute1ToWritebackType;
se : side_effect_type;
complete : std_ulogic;
exception : std_ulogic;
trap : std_ulogic;
new_msr : std_ulogic_vector(63 downto 0);
take_branch : std_ulogic;
direct_branch : std_ulogic;
start_mul : std_ulogic;
start_div : std_ulogic;
do_trace : std_ulogic;
fp_intr : std_ulogic;
res2_sel : std_ulogic_vector(1 downto 0);
bypass_valid : std_ulogic;
end record;
constant actions_type_init : actions_type :=
(e => Execute1ToWritebackInit, se => side_effect_init,
new_msr => (others => '0'), res2_sel => "00", others => '0');
type reg_stage1_type is record
e : Execute1ToWritebackType;
e : Execute1ToWritebackType;
se : side_effect_type;
busy: std_ulogic;
busy: std_ulogic;
terminate: std_ulogic;
intr_pending : std_ulogic;
fp_exception_next : std_ulogic;
fp_exception_next : std_ulogic;
trace_next : std_ulogic;
trace_next : std_ulogic;
prev_op : insn_type_t;
prev_op : insn_type_t;
br_taken : std_ulogic;
br_taken : std_ulogic;
oe : std_ulogic;
oe : std_ulogic;
mul_select : std_ulogic_vector(1 downto 0);
mul_select : std_ulogic_vector(1 downto 0);
res2_sel : std_ulogic_vector(1 downto 0);
spr_select : spr_id;
pmu_spr_num : std_ulogic_vector(4 downto 0);
mul_in_progress : std_ulogic;
mul_in_progress : std_ulogic;
mul_finish : std_ulogic;
mul_finish : std_ulogic;
div_in_progress : std_ulogic;
div_in_progress : std_ulogic;
cntz_in_progress : std_ulogic;
no_instr_avail : std_ulogic;
no_instr_avail : std_ulogic;
instr_dispatch : std_ulogic;
instr_dispatch : std_ulogic;
ext_interrupt : std_ulogic;
ext_interrupt : std_ulogic;
taken_branch_event : std_ulogic;
taken_branch_event : std_ulogic;
br_mispredict : std_ulogic;
br_mispredict : std_ulogic;
log_addr_spr : std_ulogic_vector(31 downto 0);
msr : std_ulogic_vector(63 downto 0);
xerc : xer_common_t;
xerc_valid : std_ulogic;
end record;
end record;
constant reg_type_init : reg_type :=
constant reg_stage1_type_init : reg_stage1_type :=
(e => Execute1ToWritebackInit,
(e => Execute1ToWritebackInit, se => side_effect_init,
busy => '0', terminate => '0', intr_pending => '0',
busy => '0',
fp_exception_next => '0', trace_next => '0', prev_op => OP_ILLEGAL, br_taken => '0',
fp_exception_next => '0', trace_next => '0', prev_op => OP_ILLEGAL, br_taken => '0',
oe => '0', mul_select => "00",
oe => '0', mul_select => "00", res2_sel => "00",
mul_in_progress => '0', mul_finish => '0', div_in_progress => '0', cntz_in_progress => '0',
spr_select => spr_id_init, pmu_spr_num => 5x"0",
mul_in_progress => '0', mul_finish => '0', div_in_progress => '0',
no_instr_avail => '0', instr_dispatch => '0', ext_interrupt => '0',
no_instr_avail => '0', instr_dispatch => '0', ext_interrupt => '0',
taken_branch_event => '0', br_mispredict => '0',
taken_branch_event => '0', br_mispredict => '0',
others => (others => '0'));
msr => 64x"0",
xerc => xerc_init, xerc_valid => '0');
type actions_type is record
type reg_stage2_type is record
e : Execute1ToWritebackType;
e : Execute1ToWritebackType;
complete : std_ulogic;
se : side_effect_type;
exception : std_ulogic;
ext_interrupt : std_ulogic;
trap : std_ulogic;
taken_branch_event : std_ulogic;
terminate : std_ulogic;
br_mispredict : std_ulogic;
write_msr : std_ulogic;
log_addr_spr : std_ulogic_vector(31 downto 0);
new_msr : std_ulogic_vector(63 downto 0);
write_xerlow : std_ulogic;
write_pmuspr : std_ulogic;
write_dec : std_ulogic;
write_loga : std_ulogic;
inc_loga : std_ulogic;
write_cfar : std_ulogic;
take_branch : std_ulogic;
direct_branch : std_ulogic;
start_mul : std_ulogic;
start_div : std_ulogic;
start_cntz : std_ulogic;
do_trace : std_ulogic;
fp_intr : std_ulogic;
icache_inval : std_ulogic;
end record;
end record;
constant actions_type_init : actions_type :=
constant reg_stage2_type_init : reg_stage2_type :=
(e => Execute1ToWritebackInit, new_msr => (others => '0'), others => '0');
(e => Execute1ToWritebackInit, se => side_effect_init,
log_addr_spr => 32x"0", others => '0');
signal ex1, ex1in : reg_type;
signal ex1, ex1in : reg_stage1_type;
signal ex2, ex2in : reg_stage2_type;
signal actions : actions_type;
signal actions : actions_type;
signal a_in, b_in, c_in : std_ulogic_vector(63 downto 0);
signal a_in, b_in, c_in : std_ulogic_vector(63 downto 0);
@ -142,7 +167,9 @@ architecture behaviour of execute1 is
signal muldiv_result: std_ulogic_vector(63 downto 0);
signal muldiv_result: std_ulogic_vector(63 downto 0);
signal shortmul_result: std_ulogic_vector(63 downto 0);
signal shortmul_result: std_ulogic_vector(63 downto 0);
signal spr_result: std_ulogic_vector(63 downto 0);
signal spr_result: std_ulogic_vector(63 downto 0);
signal ex_result: std_ulogic_vector(63 downto 0);
signal next_nia : std_ulogic_vector(63 downto 0);
signal next_nia : std_ulogic_vector(63 downto 0);
signal s1_sel : std_ulogic_vector(2 downto 0);
signal carry_32 : std_ulogic;
signal carry_32 : std_ulogic;
signal carry_64 : std_ulogic;
signal carry_64 : std_ulogic;
@ -372,7 +399,7 @@ begin
end generate;
end generate;
dbg_ctrl_out <= ctrl;
dbg_ctrl_out <= ctrl;
log_rd_addr <= ex1.log_addr_spr;
log_rd_addr <= ex2.log_addr_spr;
a_in <= e_in.read_data1;
a_in <= e_in.read_data1;
b_in <= e_in.read_data2;
b_in <= e_in.read_data2;
@ -393,15 +420,15 @@ begin
itlb_miss_resolved => ic_events.itlb_miss_resolved,
itlb_miss_resolved => ic_events.itlb_miss_resolved,
no_instr_avail => ex1.no_instr_avail,
no_instr_avail => ex1.no_instr_avail,
dispatch => ex1.instr_dispatch,
dispatch => ex1.instr_dispatch,
ext_interrupt => ex1.ext_interrupt,
ext_interrupt => ex2.ext_interrupt,
br_taken_complete => ex1.taken_branch_event,
br_taken_complete => ex2.taken_branch_event,
br_mispredict => ex1.br_mispredict,
br_mispredict => ex2.br_mispredict,
others => '0');
others => '0');
x_to_pmu.nia <= e_in.nia;
x_to_pmu.nia <= e_in.nia;
x_to_pmu.addr <= (others => '0');
x_to_pmu.addr <= (others => '0');
x_to_pmu.addr_v <= '0';
x_to_pmu.addr_v <= '0';
x_to_pmu.spr_num <= e_in.insn(20 downto 16);
x_to_pmu.spr_num <= ex1.pmu_spr_num;
x_to_pmu.spr_val <= c_in;
x_to_pmu.spr_val <= ex1.e.write_data;
x_to_pmu.run <= '1';
x_to_pmu.run <= '1';
-- XER forwarding. To avoid having to track XER hazards, we use
-- XER forwarding. To avoid having to track XER hazards, we use
@ -409,35 +436,23 @@ begin
-- (SO, OV[32] and CA[32]) are only modified by instructions that are
-- (SO, OV[32] and CA[32]) are only modified by instructions that are
-- handled here, we can just forward the result being sent to
-- handled here, we can just forward the result being sent to
-- writeback.
-- writeback.
xerc_in <= ex1.e.xerc when (ex1.e.write_xerc_enable and ex1.e.valid) = '1' else e_in.xerc;
xerc_in <= ex1.xerc when ex1.xerc_valid = '1' else e_in.xerc;
with e_in.unit select busy_out <=
with e_in.unit select busy_out <=
l_in.busy or ex1.busy or fp_in.busy when LDST,
l_in.busy or ex1.e.valid or ex1.busy or fp_in.busy when LDST,
l_in.busy or l_in.in_progress or ex1.e.valid or ex1.busy or fp_in.busy when FPU,
l_in.busy or l_in.in_progress or ex1.busy or fp_in.busy when others;
l_in.busy or l_in.in_progress or ex1.busy or fp_in.busy when others;
valid_in <= e_in.valid and not busy_out and not flush_in;
valid_in <= e_in.valid and not (busy_out or flush_in or ex1.e.redirect or ex1.e.interrupt);
terminate_out <= ex1.terminate;
-- First stage result mux
s1_sel <= e_in.result_sel when ex1.busy = '0' else "100";
-- Slow SPR read mux
with s1_sel select alu_result <=
with e_in.spr_select.sel select spr_result <=
ctrl.tb when SPRSEL_TB,
32x"0" & ctrl.tb(63 downto 32) when SPRSEL_TBU,
ctrl.dec when SPRSEL_DEC,
32x"0" & PVR_MICROWATT when SPRSEL_PVR,
log_wr_addr & ex1.log_addr_spr when SPRSEL_LOGA,
log_rd_data when SPRSEL_LOGD,
ctrl.cfar when SPRSEL_CFAR,
assemble_xer(xerc_in, ctrl.xer_low) when others;
-- Result mux
with e_in.result_sel select alu_result <=
adder_result when "000",
adder_result when "000",
logical_result when "001",
logical_result when "001",
rotator_result when "010",
rotator_result when "010",
shortmul_result when "011",
shortmul_result when "011",
pmu_to_x.spr_val when "100",
muldiv_result when "100",
spr_result when "101",
next_nia when "110",
next_nia when "110",
misc_result when others;
misc_result when others;
@ -445,22 +460,31 @@ begin
begin
begin
if rising_edge(clk) then
if rising_edge(clk) then
if rst = '1' then
if rst = '1' then
ex1 <= reg_type_init;
ex1 <= reg_stage1_type_init;
ex2 <= reg_stage2_type_init;
ctrl <= ctrl_t_init;
ctrl <= ctrl_t_init;
ctrl.msr <= (MSR_SF => '1', MSR_LE => '1', others => '0');
ctrl.msr <= (MSR_SF => '1', MSR_LE => '1', others => '0');
ex1.msr <= (MSR_SF => '1', MSR_LE => '1', others => '0');
else
else
ex1 <= ex1in;
ex1 <= ex1in;
ex2 <= ex2in;
ctrl <= ctrl_tmp;
ctrl <= ctrl_tmp;
if valid_in = '1' then
if valid_in = '1' then
report "execute " & to_hstring(e_in.nia) & " op=" & insn_type_t'image(e_in.insn_type) &
report "execute " & to_hstring(e_in.nia) & " op=" & insn_type_t'image(e_in.insn_type) &
" wr=" & to_hstring(ex1in.e.write_reg) & " we=" & std_ulogic'image(ex1in.e.write_enable) &
" wr=" & to_hstring(ex1in.e.write_reg) & " we=" & std_ulogic'image(ex1in.e.write_enable) &
" tag=" & integer'image(ex1in.e.instr_tag.tag) & std_ulogic'image(ex1in.e.instr_tag.valid);
" tag=" & integer'image(ex1in.e.instr_tag.tag) & std_ulogic'image(ex1in.e.instr_tag.valid);
end if;
end if;
-- We mustn't get stalled on a cycle where execute2 is
-- completing an instruction or generating an interrupt
if ex2.e.valid = '1' or ex2.e.interrupt = '1' then
assert (l_in.busy or fp_in.busy) = '0'
severity failure;
end if;
end if;
end if;
end if;
end if;
end process;
end process;
-- Data path for integer instructions
-- Data path for integer instructions (first execute stage)
execute1_dp: process(all)
execute1_dp: process(all)
variable a_inv : std_ulogic_vector(63 downto 0);
variable a_inv : std_ulogic_vector(63 downto 0);
variable b_or_m1 : std_ulogic_vector(63 downto 0);
variable b_or_m1 : std_ulogic_vector(63 downto 0);
@ -543,6 +567,7 @@ begin
if e_in.insn_type = OP_MOD then
if e_in.insn_type = OP_MOD then
x_to_divider.is_modulus <= '1';
x_to_divider.is_modulus <= '1';
end if;
end if;
x_to_divider.flush <= flush_in;
addend := (others => '0');
addend := (others => '0');
if e_in.insn(26) = '0' then
if e_in.insn(26) = '0' then
@ -638,7 +663,7 @@ begin
misc_result <= darn;
misc_result <= darn;
when "100" =>
when "100" =>
-- mfmsr
-- mfmsr
misc_result <= ctrl.msr;
misc_result <= ex1.msr;
when "101" =>
when "101" =>
if e_in.insn(20) = '0' then
if e_in.insn(20) = '0' then
-- mfcr
-- mfcr
@ -792,6 +817,7 @@ begin
variable illegal : std_ulogic;
variable illegal : std_ulogic;
variable privileged : std_ulogic;
variable privileged : std_ulogic;
variable slow_op : std_ulogic;
variable slow_op : std_ulogic;
variable owait : std_ulogic;
begin
begin
v := actions_type_init;
v := actions_type_init;
v.e.write_data := alu_result;
v.e.write_data := alu_result;
@ -803,12 +829,11 @@ begin
v.e.write_cr_enable := e_in.output_cr;
v.e.write_cr_enable := e_in.output_cr;
v.e.write_xerc_enable := e_in.output_xer;
v.e.write_xerc_enable := e_in.output_xer;
v.e.xerc := xerc_in;
v.e.xerc := xerc_in;
v.new_msr := ctrl.msr;
v.new_msr := ex1.msr;
v.e.write_xerc_enable := e_in.output_xer;
v.e.redir_mode := ex1.msr(MSR_IR) & not ex1.msr(MSR_PR) &
v.e.redir_mode := ctrl.msr(MSR_IR) & not ctrl.msr(MSR_PR) &
not ex1.msr(MSR_LE) & not ex1.msr(MSR_SF);
not ctrl.msr(MSR_LE) & not ctrl.msr(MSR_SF);
v.e.intr_vec := 16#700#;
v.e.intr_vec := 16#700#;
v.e.mode_32bit := not ctrl.msr(MSR_SF);
v.e.mode_32bit := not ex1.msr(MSR_SF);
v.e.instr_tag := e_in.instr_tag;
v.e.instr_tag := e_in.instr_tag;
v.e.last_nia := e_in.nia;
v.e.last_nia := e_in.nia;
v.e.br_offset := 64x"4";
v.e.br_offset := 64x"4";
@ -827,8 +852,9 @@ begin
illegal := '0';
illegal := '0';
privileged := '0';
privileged := '0';
slow_op := '0';
slow_op := '0';
owait := '0';
if ctrl.msr(MSR_PR) = '1' and instr_is_privileged(e_in.insn_type, e_in.insn) then
if ex1.msr(MSR_PR) = '1' and instr_is_privileged(e_in.insn_type, e_in.insn) then
privileged := '1';
privileged := '1';
end if;
end if;
@ -837,7 +863,7 @@ begin
illegal := '1';
illegal := '1';
end if;
end if;
v.do_trace := ctrl.msr(MSR_SE);
v.do_trace := ex1.msr(MSR_SE);
case_0: case e_in.insn_type is
case_0: case e_in.insn_type is
when OP_ILLEGAL =>
when OP_ILLEGAL =>
illegal := '1';
illegal := '1';
@ -858,7 +884,7 @@ begin
-- check bits 1-10 of the instruction to make sure it's attn
-- check bits 1-10 of the instruction to make sure it's attn
-- if not then it is illegal
-- if not then it is illegal
if e_in.insn(10 downto 1) = "0100000000" then
if e_in.insn(10 downto 1) = "0100000000" then
v.terminate := '1';
v.se.terminate := '1';
if e_in.valid = '1' then
if e_in.valid = '1' then
report "ATTN";
report "ATTN";
end if;
end if;
@ -909,10 +935,10 @@ begin
-- should never happen
-- should never happen
v.e.redirect := '1';
v.e.redirect := '1';
end if;
end if;
if ctrl.msr(MSR_BE) = '1' then
if ex1.msr(MSR_BE) = '1' then
v.do_trace := '1';
v.do_trace := '1';
end if;
end if;
v.write_cfar := '1';
v.se.write_cfar := '1';
when OP_BC =>
when OP_BC =>
-- read_data1 is CTR
-- read_data1 is CTR
-- If this instruction updates both CTR and LR, then it is
-- If this instruction updates both CTR and LR, then it is
@ -938,10 +964,10 @@ begin
v.direct_branch := '1';
v.direct_branch := '1';
v.e.br_last := '1';
v.e.br_last := '1';
v.e.br_taken := v.take_branch;
v.e.br_taken := v.take_branch;
if ctrl.msr(MSR_BE) = '1' then
if ex1.msr(MSR_BE) = '1' then
v.do_trace := '1';
v.do_trace := '1';
end if;
end if;
v.write_cfar := v.take_branch;
v.se.write_cfar := v.take_branch;
end if;
end if;
when OP_BCREG =>
when OP_BCREG =>
-- read_data1 is CTR, read_data2 is target register (CTR, LR or TAR)
-- read_data1 is CTR, read_data2 is target register (CTR, LR or TAR)
@ -964,10 +990,10 @@ begin
-- Indirect branches are never predicted taken
-- Indirect branches are never predicted taken
v.e.redirect := v.take_branch;
v.e.redirect := v.take_branch;
v.e.br_taken := v.take_branch;
v.e.br_taken := v.take_branch;
if ctrl.msr(MSR_BE) = '1' then
if ex1.msr(MSR_BE) = '1' then
v.do_trace := '1';
v.do_trace := '1';
end if;
end if;
v.write_cfar := v.take_branch;
v.se.write_cfar := v.take_branch;
end if;
end if;
when OP_RFID =>
when OP_RFID =>
@ -983,11 +1009,11 @@ begin
v.new_msr(MSR_IR) := '1';
v.new_msr(MSR_IR) := '1';
v.new_msr(MSR_DR) := '1';
v.new_msr(MSR_DR) := '1';
end if;
end if;
v.write_msr := '1';
v.se.write_msr := '1';
v.e.br_offset := b_in;
v.e.br_offset := b_in;
v.e.abs_br := '1';
v.e.abs_br := '1';
v.e.redirect := '1';
v.e.redirect := '1';
v.write_cfar := '1';
v.se.write_cfar := '1';
if HAS_FPU then
if HAS_FPU then
v.fp_intr := fp_in.exception and
v.fp_intr := fp_in.exception and
(a_in(MSR_FE0) or a_in(MSR_FE1));
(a_in(MSR_FE0) or a_in(MSR_FE1));
@ -995,8 +1021,8 @@ begin
v.do_trace := '0';
v.do_trace := '0';
when OP_CNTZ | OP_POPCNT =>
when OP_CNTZ | OP_POPCNT =>
v.res2_sel := "01";
slow_op := '1';
slow_op := '1';
v.start_cntz := '1';
when OP_ISEL =>
when OP_ISEL =>
when OP_CROP =>
when OP_CROP =>
when OP_MCRXRX =>
when OP_MCRXRX =>
@ -1010,14 +1036,19 @@ begin
end if;
end if;
elsif e_in.spr_select.valid = '1' then
elsif e_in.spr_select.valid = '1' then
if e_in.valid = '1' then
if e_in.valid = '1' then
report "MFSPR to SPR " & integer'image(decode_spr_num(e_in.insn)) &
report "MFSPR to slow SPR " & integer'image(decode_spr_num(e_in.insn));
"=" & to_hstring(spr_result);
end if;
slow_op := '1';
if e_in.spr_select.ispmu = '0' then
case e_in.spr_select.sel is
when SPRSEL_LOGD =>
v.se.inc_loga := '1';
when others =>
end case;
v.res2_sel := "10";
else
v.res2_sel := "11";
end if;
end if;
case e_in.spr_select.sel is
when SPRSEL_LOGD =>
v.inc_loga := '1';
when others =>
end case;
else
else
-- mfspr from unimplemented SPRs should be a nop in
-- mfspr from unimplemented SPRs should be a nop in
-- supervisor mode and a program interrupt for user mode
-- supervisor mode and a program interrupt for user mode
@ -1025,7 +1056,7 @@ begin
report "MFSPR to SPR " & integer'image(decode_spr_num(e_in.insn)) &
report "MFSPR to SPR " & integer'image(decode_spr_num(e_in.insn)) &
" invalid";
" invalid";
end if;
end if;
if ctrl.msr(MSR_PR) = '1' then
if ex1.msr(MSR_PR) = '1' then
illegal := '1';
illegal := '1';
end if;
end if;
end if;
end if;
@ -1033,7 +1064,7 @@ begin
when OP_MFCR =>
when OP_MFCR =>
when OP_MTCRF =>
when OP_MTCRF =>
when OP_MTMSRD =>
when OP_MTMSRD =>
v.write_msr := '1';
v.se.write_msr := '1';
if e_in.insn(16) = '1' then
if e_in.insn(16) = '1' then
-- just update EE and RI
-- just update EE and RI
v.new_msr(MSR_EE) := c_in(MSR_EE);
v.new_msr(MSR_EE) := c_in(MSR_EE);
@ -1062,7 +1093,7 @@ begin
report "MTSPR to SPR " & integer'image(decode_spr_num(e_in.insn)) &
report "MTSPR to SPR " & integer'image(decode_spr_num(e_in.insn)) &
"=" & to_hstring(c_in);
"=" & to_hstring(c_in);
end if;
end if;
v.write_pmuspr := e_in.spr_select.ispmu;
v.se.write_pmuspr := e_in.spr_select.ispmu;
if e_in.spr_select.valid = '1' and e_in.spr_select.ispmu = '0' then
if e_in.spr_select.valid = '1' and e_in.spr_select.ispmu = '0' then
case e_in.spr_select.sel is
case e_in.spr_select.sel is
when SPRSEL_XER =>
when SPRSEL_XER =>
@ -1071,17 +1102,17 @@ begin
v.e.xerc.ca := c_in(63-34);
v.e.xerc.ca := c_in(63-34);
v.e.xerc.ov32 := c_in(63-44);
v.e.xerc.ov32 := c_in(63-44);
v.e.xerc.ca32 := c_in(63-45);
v.e.xerc.ca32 := c_in(63-45);
v.write_xerlow := '1';
v.se.write_xerlow := '1';
when SPRSEL_DEC =>
when SPRSEL_DEC =>
v.write_dec := '1';
v.se.write_dec := '1';
when SPRSEL_LOGA =>
when SPRSEL_LOGA =>
v.write_loga := '1';
v.se.write_loga := '1';
when others =>
when others =>
end case;
end case;
elsif is_fast_spr(e_in.write_reg) = '0' then
elsif is_fast_spr(e_in.write_reg) = '0' then
-- mtspr to unimplemented SPRs should be a nop in
-- mtspr to unimplemented SPRs should be a nop in
-- supervisor mode and a program interrupt for user mode
-- supervisor mode and a program interrupt for user mode
if ctrl.msr(MSR_PR) = '1' then
if ex1.msr(MSR_PR) = '1' then
illegal := '1';
illegal := '1';
end if;
end if;
end if;
end if;
@ -1095,7 +1126,7 @@ begin
v.e.redirect := '1';
v.e.redirect := '1';
when OP_ICBI =>
when OP_ICBI =>
v.icache_inval := '1';
v.se.icache_inval := '1';
when OP_MUL_L64 =>
when OP_MUL_L64 =>
if HAS_SHORT_MULT and e_in.insn(26) = '1' and
if HAS_SHORT_MULT and e_in.insn(26) = '1' and
@ -1109,15 +1140,18 @@ begin
-- Use standard multiplier
-- Use standard multiplier
v.start_mul := '1';
v.start_mul := '1';
slow_op := '1';
slow_op := '1';
owait := '1';
end if;
end if;
when OP_MUL_H64 | OP_MUL_H32 =>
when OP_MUL_H64 | OP_MUL_H32 =>
v.start_mul := '1';
v.start_mul := '1';
slow_op := '1';
slow_op := '1';
owait := '1';
when OP_DIV | OP_DIVE | OP_MOD =>
when OP_DIV | OP_DIVE | OP_MOD =>
v.start_div := '1';
v.start_div := '1';
slow_op := '1';
slow_op := '1';
owait := '1';
when OP_FETCH_FAILED =>
when OP_FETCH_FAILED =>
-- Handling an ITLB miss doesn't count as having executed an instruction
-- Handling an ITLB miss doesn't count as having executed an instruction
@ -1147,7 +1181,7 @@ begin
report "illegal instruction";
report "illegal instruction";
end if;
end if;
elsif HAS_FPU and ctrl.msr(MSR_FP) = '0' and e_in.fac = FPU then
elsif HAS_FPU and ex1.msr(MSR_FP) = '0' and e_in.fac = FPU then
-- generate a floating-point unavailable interrupt
-- generate a floating-point unavailable interrupt
v.exception := '1';
v.exception := '1';
v.e.intr_vec := 16#800#;
v.e.intr_vec := 16#800#;
@ -1157,26 +1191,33 @@ begin
end if;
end if;
if e_in.unit = ALU then
if e_in.unit = ALU then
v.complete := e_in.valid and not v.exception and not slow_op;
v.complete := e_in.valid and not v.exception and not owait;
v.bypass_valid := e_in.valid and not v.exception and not slow_op;
end if;
end if;
actions <= v;
actions <= v;
end process;
end process;
-- First execute stage
execute1_1: process(all)
execute1_1: process(all)
variable v : reg_type;
variable v : reg_stage1_type;
variable overflow : std_ulogic;
variable overflow : std_ulogic;
variable lv : Execute1ToLoadstore1Type;
variable lv : Execute1ToLoadstore1Type;
variable irq_valid : std_ulogic;
variable irq_valid : std_ulogic;
variable exception : std_ulogic;
variable exception : std_ulogic;
variable fv : Execute1ToFPUType;
variable fv : Execute1ToFPUType;
variable go : std_ulogic;
variable go : std_ulogic;
variable bypass_valid : std_ulogic;
begin
begin
v := ex1;
v := ex1;
if ex1.busy = '0' then
if (ex1.busy or l_in.busy or fp_in.busy) = '0' then
v.e := actions.e;
v.e := actions.e;
v.e.valid := '0';
v.oe := e_in.oe;
v.oe := e_in.oe;
v.spr_select := e_in.spr_select;
v.pmu_spr_num := e_in.insn(20 downto 16);
v.mul_select := e_in.sub_select(1 downto 0);
v.mul_select := e_in.sub_select(1 downto 0);
v.se := side_effect_init;
end if;
end if;
lv := Execute1ToLoadstore1Init;
lv := Execute1ToLoadstore1Init;
@ -1184,33 +1225,13 @@ begin
x_to_multiply.valid <= '0';
x_to_multiply.valid <= '0';
x_to_divider.valid <= '0';
x_to_divider.valid <= '0';
v.mul_in_progress := '0';
v.div_in_progress := '0';
v.cntz_in_progress := '0';
v.mul_finish := '0';
v.ext_interrupt := '0';
v.ext_interrupt := '0';
v.taken_branch_event := '0';
v.taken_branch_event := '0';
v.br_mispredict := '0';
v.br_mispredict := '0';
v.busy := '0';
bypass_valid := '0';
x_to_pmu.mfspr <= '0';
irq_valid := ex1.msr(MSR_EE) and (pmu_to_x.intr or ctrl.dec(63) or ext_irq_in);
x_to_pmu.mtspr <= '0';
x_to_pmu.tbbits(3) <= ctrl.tb(63 - 47);
x_to_pmu.tbbits(2) <= ctrl.tb(63 - 51);
x_to_pmu.tbbits(1) <= ctrl.tb(63 - 55);
x_to_pmu.tbbits(0) <= ctrl.tb(63 - 63);
x_to_pmu.pmm_msr <= ctrl.msr(MSR_PMM);
x_to_pmu.pr_msr <= ctrl.msr(MSR_PR);
ctrl_tmp <= ctrl;
-- FIXME: run at 512MHz not core freq
ctrl_tmp.tb <= std_ulogic_vector(unsigned(ctrl.tb) + 1);
ctrl_tmp.dec <= std_ulogic_vector(unsigned(ctrl.dec) - 1);
irq_valid := ctrl.msr(MSR_EE) and (pmu_to_x.intr or ctrl.dec(63) or ext_irq_in);
v.terminate := '0';
icache_inval <= '0';
v.busy := '0';
-- Next insn adder used in a couple of places
-- Next insn adder used in a couple of places
next_nia <= std_ulogic_vector(unsigned(e_in.nia) + 4);
next_nia <= std_ulogic_vector(unsigned(e_in.nia) + 4);
@ -1223,19 +1244,14 @@ begin
do_popcnt <= '1' when e_in.insn_type = OP_POPCNT else '0';
do_popcnt <= '1' when e_in.insn_type = OP_POPCNT else '0';
if ex1.intr_pending = '1' then
v.e.srr1 := ex1.e.srr1;
v.e.intr_vec := ex1.e.intr_vec;
end if;
if valid_in = '1' then
if valid_in = '1' then
v.prev_op := e_in.insn_type;
v.prev_op := e_in.insn_type;
end if;
end if;
-- Determine if there is any interrupt to be taken
-- Determine if there is any interrupt to be taken
-- before/instead of executing this instruction
-- before/instead of executing this instruction
exception := ex1.intr_pending or (valid_in and actions.exception);
exception := valid_in and actions.exception;
if valid_in = '1' and e_in.second = '0' and ex1.intr_pending = '0' then
if valid_in = '1' and e_in.second = '0' then
if HAS_FPU and ex1.fp_exception_next = '1' then
if HAS_FPU and ex1.fp_exception_next = '1' then
-- This is used for FP-type program interrupts that
-- This is used for FP-type program interrupts that
-- become pending due to MSR[FE0,FE1] changing from 00 to non-zero.
-- become pending due to MSR[FE0,FE1] changing from 00 to non-zero.
@ -1278,54 +1294,37 @@ begin
end if;
end if;
end if;
end if;
if exception = '1' and l_in.in_progress = '1' then
-- We can't send this interrupt to writeback yet because there are
-- still instructions in loadstore1 that haven't completed.
v.intr_pending := '1';
v.busy := '1';
end if;
v.no_instr_avail := not (e_in.valid or l_in.busy or l_in.in_progress or ex1.busy or fp_in.busy);
v.no_instr_avail := not (e_in.valid or l_in.busy or l_in.in_progress or
ex1.busy or fp_in.busy);
go := valid_in and not exception;
go := valid_in and not exception;
v.instr_dispatch := go;
v.instr_dispatch := go;
if go = '1' then
if go = '1' then
v.se := actions.se;
v.e.valid := actions.complete;
v.e.valid := actions.complete;
bypass_valid := actions.bypass_valid;
v.taken_branch_event := actions.take_branch;
v.taken_branch_event := actions.take_branch;
v.br_taken := actions.take_branch;
v.br_taken := actions.take_branch;
v.trace_next := actions.do_trace;
v.trace_next := actions.do_trace;
v.fp_exception_next := actions.fp_intr;
v.fp_exception_next := actions.fp_intr;
v.cntz_in_progress := actions.start_cntz;
v.res2_sel := actions.res2_sel;
v.msr := actions.new_msr;
if actions.write_msr = '1' then
ctrl_tmp.msr <= actions.new_msr;
end if;
if actions.write_xerlow = '1' then
ctrl_tmp.xer_low <= c_in(17 downto 0);
end if;
if actions.write_dec = '1' then
ctrl_tmp.dec <= c_in;
end if;
if actions.write_cfar = '1' then
ctrl_tmp.cfar <= e_in.nia;
end if;
if actions.write_loga = '1' then
v.log_addr_spr := c_in(31 downto 0);
elsif actions.inc_loga = '1' then
v.log_addr_spr := std_ulogic_vector(unsigned(ex1.log_addr_spr) + 1);
end if;
x_to_pmu.mtspr <= actions.write_pmuspr;
icache_inval <= actions.icache_inval;
x_to_multiply.valid <= actions.start_mul;
x_to_multiply.valid <= actions.start_mul;
v.mul_in_progress := actions.start_mul;
v.mul_in_progress := actions.start_mul;
x_to_divider.valid <= actions.start_div;
x_to_divider.valid <= actions.start_div;
v.div_in_progress := actions.start_div;
v.div_in_progress := actions.start_div;
v.terminate := actions.terminate;
v.br_mispredict := v.e.redirect and actions.direct_branch;
v.br_mispredict := v.e.redirect and actions.direct_branch;
v.busy := actions.start_cntz or actions.start_mul or actions.start_div;
exception := actions.trap;
exception := actions.trap;
-- Go busy while division is happening because the
-- divider is not pipelined. Also go busy while a
-- multiply is happening in order to stop following
-- instructions from using the wrong XER value
-- (and for simplicity in the OE=0 case).
v.busy := actions.start_div or actions.start_mul;
-- instruction for other units, i.e. LDST
-- instruction for other units, i.e. LDST
if e_in.unit = LDST then
if e_in.unit = LDST then
lv.valid := '1';
lv.valid := '1';
@ -1335,86 +1334,74 @@ begin
end if;
end if;
end if;
end if;
-- The following cases all occur when ex1.busy = 1 and therefore
if ex1.div_in_progress = '1' then
-- valid_in = 0. Hence they don't happen in the same cycle as any of
v.div_in_progress := not divider_to_x.valid;
-- the cases above which depend on valid_in = 1.
v.busy := not divider_to_x.valid;
if ex1.cntz_in_progress = '1' then
if divider_to_x.valid = '1' and ex1.oe = '1' then
-- cnt[lt]z and popcnt* always take two cycles
v.e.xerc.ov := divider_to_x.overflow;
v.e.valid := '1';
v.e.xerc.ov32 := divider_to_x.overflow;
v.e.write_data := countbits_result;
if divider_to_x.overflow = '1' then
end if;
v.e.xerc.so := '1';
if ex1.div_in_progress = '1' then
if divider_to_x.valid = '1' then
v.e.write_data := muldiv_result;
overflow := divider_to_x.overflow;
-- We must test oe because the RC update code in writeback
-- will use the xerc value to set CR0:SO so we must not clobber
-- xerc if OE wasn't set.
if ex1.oe = '1' then
v.e.xerc.ov := overflow;
v.e.xerc.ov32 := overflow;
if overflow = '1' then
v.e.xerc.so := '1';
end if;
end if;
end if;
v.e.valid := '1';
end if;
else
v.e.valid := divider_to_x.valid;
v.busy := '1';
v.e.write_data := alu_result;
v.div_in_progress := '1';
bypass_valid := v.e.valid;
end if;
end if;
end if;
if ex1.mul_in_progress = '1' then
if ex1.mul_in_progress = '1' then
if multiply_to_x.valid = '1' then
v.mul_in_progress := not multiply_to_x.valid;
v.e.write_data := muldiv_result;
v.mul_finish := multiply_to_x.valid and ex1.oe;
if ex1.oe = '1' then
v.e.valid := multiply_to_x.valid and not ex1.oe;
-- have to wait until next cycle for overflow indication
v.busy := not v.e.valid;
v.mul_finish := '1';
v.e.write_data := alu_result;
v.busy := '1';
bypass_valid := v.e.valid;
else
v.e.valid := '1';
end if;
else
v.busy := '1';
v.mul_in_progress := '1';
end if;
end if;
end if;
if ex1.mul_finish = '1' then
if ex1.mul_finish = '1' then
v.mul_finish := '0';
v.e.xerc.ov := multiply_to_x.overflow;
v.e.xerc.ov := multiply_to_x.overflow;
v.e.xerc.ov32 := multiply_to_x.overflow;
v.e.xerc.ov32 := multiply_to_x.overflow;
if multiply_to_x.overflow = '1' then
if multiply_to_x.overflow = '1' then
v.e.xerc.so := '1';
v.e.xerc.so := '1';
end if;
end if;
v.e.valid := '1';
v.e.valid := '1';
end if;
end if;
v.e.interrupt := exception and not (l_in.in_progress or l_in.interrupt);
if v.e.write_xerc_enable = '1' and v.e.valid = '1' then
if v.e.interrupt = '1' then
v.xerc := v.e.xerc;
v.intr_pending := '0';
v.xerc_valid := '1';
end if;
end if;
if interrupt_in = '1' then
if (ex1.busy or l_in.busy or fp_in.busy) = '0' then
ctrl_tmp.msr(MSR_SF) <= '1';
v.e.interrupt := exception;
ctrl_tmp.msr(MSR_EE) <= '0';
end if;
ctrl_tmp.msr(MSR_PR) <= '0';
if v.e.valid = '0' then
ctrl_tmp.msr(MSR_SE) <= '0';
v.e.redirect := '0';
ctrl_tmp.msr(MSR_BE) <= '0';
v.e.br_last := '0';
ctrl_tmp.msr(MSR_FP) <= '0';
end if;
ctrl_tmp.msr(MSR_FE0) <= '0';
if flush_in = '1' then
ctrl_tmp.msr(MSR_FE1) <= '0';
v.e.valid := '0';
ctrl_tmp.msr(MSR_IR) <= '0';
v.e.interrupt := '0';
ctrl_tmp.msr(MSR_DR) <= '0';
v.e.redirect := '0';
ctrl_tmp.msr(MSR_RI) <= '0';
v.e.br_last := '0';
ctrl_tmp.msr(MSR_LE) <= '1';
v.busy := '0';
v.div_in_progress := '0';
v.mul_in_progress := '0';
v.mul_finish := '0';
v.xerc_valid := '0';
end if;
if flush_in = '1' or interrupt_in = '1' then
v.msr := ctrl_tmp.msr;
end if;
if interrupt_in = '1' then
v.trace_next := '0';
v.trace_next := '0';
v.fp_exception_next := '0';
v.fp_exception_next := '0';
v.intr_pending := '0';
end if;
end if;
bypass_data.tag.valid <= v.e.write_enable and v.e.valid;
bypass_data.tag.valid <= v.e.write_enable and bypass_valid;
bypass_data.tag.tag <= v.e.instr_tag.tag;
bypass_data.tag.tag <= v.e.instr_tag.tag;
bypass_data.data <= v.e.write_data;
bypass_data.data <= alu_result;
bypass_cr_data.tag.valid <= v.e.write_cr_enable and v.e.valid;
bypass_cr_data.tag.valid <= v.e.write_cr_enable and bypass_valid;
bypass_cr_data.tag.tag <= v.e.instr_tag.tag;
bypass_cr_data.tag.tag <= v.e.instr_tag.tag;
bypass_cr_data.data <= v.e.write_cr_data;
bypass_cr_data.data <= v.e.write_cr_data;
@ -1427,7 +1414,7 @@ begin
lv.data := c_in;
lv.data := c_in;
lv.write_reg := e_in.write_reg;
lv.write_reg := e_in.write_reg;
lv.length := e_in.data_len;
lv.length := e_in.data_len;
lv.byte_reverse := e_in.byte_reverse xnor ctrl.msr(MSR_LE);
lv.byte_reverse := e_in.byte_reverse xnor ex1.msr(MSR_LE);
lv.sign_extend := e_in.sign_extend;
lv.sign_extend := e_in.sign_extend;
lv.update := e_in.update;
lv.update := e_in.update;
lv.xerc := xerc_in;
lv.xerc := xerc_in;
@ -1439,9 +1426,9 @@ begin
e_in.insn(5 downto 1) = "10101" then
e_in.insn(5 downto 1) = "10101" then
lv.ci := '1';
lv.ci := '1';
end if;
end if;
lv.virt_mode := ctrl.msr(MSR_DR);
lv.virt_mode := ex1.msr(MSR_DR);
lv.priv_mode := not ctrl.msr(MSR_PR);
lv.priv_mode := not ex1.msr(MSR_PR);
lv.mode_32bit := not ctrl.msr(MSR_SF);
lv.mode_32bit := not ex1.msr(MSR_SF);
lv.is_32bit := e_in.is_32bit;
lv.is_32bit := e_in.is_32bit;
lv.repeat := e_in.repeat;
lv.repeat := e_in.repeat;
lv.second := e_in.second;
lv.second := e_in.second;
@ -1452,7 +1439,7 @@ begin
fv.insn := e_in.insn;
fv.insn := e_in.insn;
fv.itag := e_in.instr_tag;
fv.itag := e_in.instr_tag;
fv.single := e_in.is_32bit;
fv.single := e_in.is_32bit;
fv.fe_mode := ctrl.msr(MSR_FE0) & ctrl.msr(MSR_FE1);
fv.fe_mode := ex1.msr(MSR_FE0) & ex1.msr(MSR_FE1);
fv.fra := a_in;
fv.fra := a_in;
fv.frb := b_in;
fv.frb := b_in;
fv.frc := c_in;
fv.frc := c_in;
@ -1465,19 +1452,124 @@ begin
-- update outputs
-- update outputs
l_out <= lv;
l_out <= lv;
e_out <= ex1.e;
fp_out <= fv;
if ex1.e.valid = '0' then
irq_valid_log <= irq_valid;
e_out.write_enable <= '0';
end process;
e_out.write_cr_enable <= '0';
e_out.write_xerc_enable <= '0';
-- Slow SPR read mux
e_out.redirect <= '0';
with ex1.spr_select.sel select spr_result <=
e_out.br_last <= '0';
ctrl.tb when SPRSEL_TB,
32x"0" & ctrl.tb(63 downto 32) when SPRSEL_TBU,
ctrl.dec when SPRSEL_DEC,
32x"0" & PVR_MICROWATT when SPRSEL_PVR,
log_wr_addr & ex2.log_addr_spr when SPRSEL_LOGA,
log_rd_data when SPRSEL_LOGD,
ctrl.cfar when SPRSEL_CFAR,
assemble_xer(ex1.e.xerc, ctrl.xer_low) when others;
-- Second stage result mux
with ex1.res2_sel select ex_result <=
countbits_result when "01",
spr_result when "10",
pmu_to_x.spr_val when "11",
ex1.e.write_data when others;
-- Second execute stage control
execute2_1: process(all)
variable v : reg_stage2_type;
variable overflow : std_ulogic;
variable lv : Execute1ToLoadstore1Type;
variable fv : Execute1ToFPUType;
variable k : integer;
variable go : std_ulogic;
begin
v := ex2;
if (l_in.busy or fp_in.busy) = '0' then
v.e := ex1.e;
v.se := ex1.se;
v.e.write_data := ex_result;
v.ext_interrupt := ex1.ext_interrupt;
v.taken_branch_event := ex1.taken_branch_event;
v.br_mispredict := ex1.br_mispredict;
end if;
ctrl_tmp <= ctrl;
-- FIXME: run at 512MHz not core freq
ctrl_tmp.tb <= std_ulogic_vector(unsigned(ctrl.tb) + 1);
ctrl_tmp.dec <= std_ulogic_vector(unsigned(ctrl.dec) - 1);
x_to_pmu.mfspr <= '0';
x_to_pmu.mtspr <= '0';
x_to_pmu.tbbits(3) <= ctrl.tb(63 - 47);
x_to_pmu.tbbits(2) <= ctrl.tb(63 - 51);
x_to_pmu.tbbits(1) <= ctrl.tb(63 - 55);
x_to_pmu.tbbits(0) <= ctrl.tb(63 - 63);
x_to_pmu.pmm_msr <= ctrl.msr(MSR_PMM);
x_to_pmu.pr_msr <= ctrl.msr(MSR_PR);
if v.e.valid = '0' or flush_in = '1' then
v.e.write_enable := '0';
v.e.write_cr_enable := '0';
v.e.write_xerc_enable := '0';
v.e.redirect := '0';
v.e.br_last := '0';
v.se := side_effect_init;
v.taken_branch_event := '0';
v.br_mispredict := '0';
end if;
if flush_in = '1' then
v.e.valid := '0';
v.e.interrupt := '0';
v.ext_interrupt := '0';
end if;
if (l_in.busy or fp_in.busy) = '0' then
if ex1.se.write_msr = '1' then
ctrl_tmp.msr <= ex1.msr;
end if;
if ex1.se.write_xerlow = '1' then
ctrl_tmp.xer_low <= ex1.e.write_data(17 downto 0);
end if;
if ex1.se.write_dec = '1' then
ctrl_tmp.dec <= ex1.e.write_data;
end if;
if ex1.se.write_cfar = '1' then
ctrl_tmp.cfar <= ex1.e.last_nia;
end if;
if ex1.se.write_loga = '1' then
v.log_addr_spr := ex1.e.write_data(31 downto 0);
elsif ex1.se.inc_loga = '1' then
v.log_addr_spr := std_ulogic_vector(unsigned(ex2.log_addr_spr) + 1);
end if;
x_to_pmu.mtspr <= ex1.se.write_pmuspr;
end if;
end if;
if interrupt_in = '1' then
ctrl_tmp.msr(MSR_SF) <= '1';
ctrl_tmp.msr(MSR_EE) <= '0';
ctrl_tmp.msr(MSR_PR) <= '0';
ctrl_tmp.msr(MSR_SE) <= '0';
ctrl_tmp.msr(MSR_BE) <= '0';
ctrl_tmp.msr(MSR_FP) <= '0';
ctrl_tmp.msr(MSR_FE0) <= '0';
ctrl_tmp.msr(MSR_FE1) <= '0';
ctrl_tmp.msr(MSR_IR) <= '0';
ctrl_tmp.msr(MSR_DR) <= '0';
ctrl_tmp.msr(MSR_RI) <= '0';
ctrl_tmp.msr(MSR_LE) <= '1';
end if;
-- Update registers
ex2in <= v;
-- update outputs
e_out <= ex2.e;
e_out.msr <= msr_copy(ctrl.msr);
e_out.msr <= msr_copy(ctrl.msr);
fp_out <= fv;
exception_log <= exception;
terminate_out <= ex2.se.terminate;
irq_valid_log <= irq_valid;
icache_inval <= ex2.se.icache_inval;
exception_log <= v.e.interrupt;
end process;
end process;
e1_log: if LOG_LENGTH > 0 generate
e1_log: if LOG_LENGTH > 0 generate
@ -1492,9 +1584,9 @@ begin
irq_valid_log &
irq_valid_log &
interrupt_in &
interrupt_in &
"000" &
"000" &
ex1.e.write_enable &
ex2.e.write_enable &
ex1.e.valid &
ex2.e.valid &
((ex1.e.redirect and ex1.e.valid) or ex1.e.interrupt) &
(ex2.e.redirect or ex2.e.interrupt) &
ex1.busy &
ex1.busy &
flush_in;
flush_in;
end if;
end if;