You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
369 lines
14 KiB
369 lines
14 KiB
library ieee; |
|
use ieee.std_logic_1164.all; |
|
use ieee.numeric_std.all; |
|
|
|
library work; |
|
use work.common.all; |
|
use work.decode_types.all; |
|
|
|
entity pmu is |
|
port ( |
|
clk : in std_ulogic; |
|
rst : in std_ulogic; |
|
p_in : in Execute1ToPMUType; |
|
p_out : out PMUToExecute1Type |
|
); |
|
end entity pmu; |
|
|
|
architecture behaviour of pmu is |
|
|
|
-- MMCR0 bit numbers |
|
constant MMCR0_FC : integer := 63 - 32; |
|
constant MMCR0_FCS : integer := 63 - 33; |
|
constant MMCR0_FCP : integer := 63 - 34; |
|
constant MMCR0_FCM1 : integer := 63 - 35; |
|
constant MMCR0_FCM0 : integer := 63 - 36; |
|
constant MMCR0_PMAE : integer := 63 - 37; |
|
constant MMCR0_FCECE : integer := 63 - 38; |
|
constant MMCR0_TBSEL : integer := 63 - 40; |
|
constant MMCR0_TBEE : integer := 63 - 41; |
|
constant MMCR0_BHRBA : integer := 63 - 42; |
|
constant MMCR0_EBE : integer := 63 - 43; |
|
constant MMCR0_PMCC : integer := 63 - 45; |
|
constant MMCR0_PMC1CE : integer := 63 - 48; |
|
constant MMCR0_PMCjCE : integer := 63 - 49; |
|
constant MMCR0_TRIGGER : integer := 63 - 50; |
|
constant MMCR0_FCPC : integer := 63 - 51; |
|
constant MMCR0_PMAQ : integer := 63 - 52; |
|
constant MMCR0_PMCCEXT : integer := 63 - 54; |
|
constant MMCR0_CC56RUN : integer := 63 - 55; |
|
constant MMCR0_PMAO : integer := 63 - 56; |
|
constant MMCR0_FC1_4 : integer := 63 - 58; |
|
constant MMCR0_FC5_6 : integer := 63 - 59; |
|
constant MMCR0_FC1_4W : integer := 63 - 62; |
|
|
|
-- MMCR2 bit numbers |
|
constant MMCR2_FC0S : integer := 63 - 0; |
|
constant MMCR2_FC0P0 : integer := 63 - 1; |
|
constant MMCR2_FC0M1 : integer := 63 - 3; |
|
constant MMCR2_FC0M0 : integer := 63 - 4; |
|
constant MMCR2_FC0WAIT : integer := 63 - 5; |
|
constant MMCR2_FC1S : integer := 54 - 0; |
|
constant MMCR2_FC1P0 : integer := 54 - 1; |
|
constant MMCR2_FC1M1 : integer := 54 - 3; |
|
constant MMCR2_FC1M0 : integer := 54 - 4; |
|
constant MMCR2_FC1WAIT : integer := 54 - 5; |
|
constant MMCR2_FC2S : integer := 45 - 0; |
|
constant MMCR2_FC2P0 : integer := 45 - 1; |
|
constant MMCR2_FC2M1 : integer := 45 - 3; |
|
constant MMCR2_FC2M0 : integer := 45 - 4; |
|
constant MMCR2_FC2WAIT : integer := 45 - 5; |
|
constant MMCR2_FC3S : integer := 36 - 0; |
|
constant MMCR2_FC3P0 : integer := 36 - 1; |
|
constant MMCR2_FC3M1 : integer := 36 - 3; |
|
constant MMCR2_FC3M0 : integer := 36 - 4; |
|
constant MMCR2_FC3WAIT : integer := 36 - 5; |
|
constant MMCR2_FC4S : integer := 27 - 0; |
|
constant MMCR2_FC4P0 : integer := 27 - 1; |
|
constant MMCR2_FC4M1 : integer := 27 - 3; |
|
constant MMCR2_FC4M0 : integer := 27 - 4; |
|
constant MMCR2_FC4WAIT : integer := 27 - 5; |
|
constant MMCR2_FC5S : integer := 18 - 0; |
|
constant MMCR2_FC5P0 : integer := 18 - 1; |
|
constant MMCR2_FC5M1 : integer := 18 - 3; |
|
constant MMCR2_FC5M0 : integer := 18 - 4; |
|
constant MMCR2_FC5WAIT : integer := 18 - 5; |
|
constant MMCR2_FC6S : integer := 9 - 0; |
|
constant MMCR2_FC6P0 : integer := 9 - 1; |
|
constant MMCR2_FC6M1 : integer := 9 - 3; |
|
constant MMCR2_FC6M0 : integer := 9 - 4; |
|
constant MMCR2_FC6WAIT : integer := 9 - 5; |
|
|
|
-- MMCRA bit numbers |
|
constant MMCRA_TECX : integer := 63 - 36; |
|
constant MMCRA_TECM : integer := 63 - 44; |
|
constant MMCRA_TECE : integer := 63 - 47; |
|
constant MMCRA_TS : integer := 63 - 51; |
|
constant MMCRA_TE : integer := 63 - 55; |
|
constant MMCRA_ES : integer := 63 - 59; |
|
constant MMCRA_SM : integer := 63 - 62; |
|
constant MMCRA_SE : integer := 63 - 63; |
|
|
|
-- SIER bit numbers |
|
constant SIER_SAMPPR : integer := 63 - 38; |
|
constant SIER_SIARV : integer := 63 - 41; |
|
constant SIER_SDARV : integer := 63 - 42; |
|
constant SIER_TE : integer := 63 - 43; |
|
constant SIER_SITYPE : integer := 63 - 48; |
|
constant SIER_SICACHE : integer := 63 - 51; |
|
constant SIER_SITAKBR : integer := 63 - 52; |
|
constant SIER_SIMISPR : integer := 63 - 53; |
|
constant SIER_SIMISPRI : integer := 63 - 55; |
|
constant SIER_SIDERAT : integer := 63 - 56; |
|
constant SIER_SIDAXL : integer := 63 - 59; |
|
constant SIER_SIDSAI : integer := 63 - 62; |
|
constant SIER_SICMPL : integer := 63 - 63; |
|
|
|
type pmc_array is array(1 to 6) of std_ulogic_vector(31 downto 0); |
|
signal pmcs : pmc_array; |
|
signal mmcr0 : std_ulogic_vector(31 downto 0); |
|
signal mmcr1 : std_ulogic_vector(63 downto 0); |
|
signal mmcr2 : std_ulogic_vector(63 downto 0); |
|
signal mmcra : std_ulogic_vector(63 downto 0); |
|
signal siar : std_ulogic_vector(63 downto 0); |
|
signal sdar : std_ulogic_vector(63 downto 0); |
|
signal sier : std_ulogic_vector(63 downto 0); |
|
|
|
signal doinc : std_ulogic_vector(1 to 6); |
|
signal doalert : std_ulogic; |
|
signal doevent : std_ulogic; |
|
|
|
signal prev_tb : std_ulogic_vector(3 downto 0); |
|
|
|
begin |
|
-- mfspr mux |
|
with p_in.spr_num(3 downto 0) select p_out.spr_val <= |
|
32x"0" & pmcs(1) when "0011", |
|
32x"0" & pmcs(2) when "0100", |
|
32x"0" & pmcs(3) when "0101", |
|
32x"0" & pmcs(4) when "0110", |
|
32x"0" & pmcs(5) when "0111", |
|
32x"0" & pmcs(6) when "1000", |
|
32x"0" & mmcr0 when "1011", |
|
mmcr1 when "1110", |
|
mmcr2 when "0001", |
|
mmcra when "0010", |
|
siar when "1100", |
|
sdar when "1101", |
|
sier when "0000", |
|
64x"0" when others; |
|
|
|
p_out.intr <= mmcr0(MMCR0_PMAO); |
|
|
|
pmu_1: process(clk) |
|
begin |
|
if rising_edge(clk) then |
|
if rst = '1' then |
|
mmcr0 <= 32x"80000000"; |
|
else |
|
for i in 1 to 6 loop |
|
if p_in.mtspr = '1' and to_integer(unsigned(p_in.spr_num(3 downto 0))) = i + 2 then |
|
pmcs(i) <= p_in.spr_val(31 downto 0); |
|
elsif doinc(i) = '1' then |
|
pmcs(i) <= std_ulogic_vector(unsigned(pmcs(i)) + 1); |
|
end if; |
|
end loop; |
|
if p_in.mtspr = '1' and p_in.spr_num(3 downto 0) = "1011" then |
|
mmcr0 <= p_in.spr_val(31 downto 0); |
|
mmcr0(MMCR0_BHRBA) <= '0'; -- no BHRB yet |
|
mmcr0(MMCR0_EBE) <= '0'; -- no EBBs yet |
|
else |
|
if doalert = '1' then |
|
mmcr0(MMCR0_PMAE) <= '0'; |
|
mmcr0(MMCR0_PMAO) <= '1'; |
|
mmcr0(MMCR0_PMAQ) <= '0'; |
|
end if; |
|
if doevent = '1' and mmcr0(MMCR0_FCECE) = '1' and mmcr0(MMCR0_TRIGGER) = '0' then |
|
mmcr0(MMCR0_FC) <= '1'; |
|
end if; |
|
if (doevent = '1' or pmcs(1)(31) = '1') and mmcr0(MMCR0_TRIGGER) = '1' then |
|
mmcr0(MMCR0_TRIGGER) <= '0'; |
|
end if; |
|
end if; |
|
if p_in.mtspr = '1' and p_in.spr_num(3 downto 0) = "1110" then |
|
mmcr1 <= p_in.spr_val; |
|
end if; |
|
if p_in.mtspr = '1' and p_in.spr_num(3 downto 0) = "0001" then |
|
mmcr2 <= p_in.spr_val; |
|
end if; |
|
if p_in.mtspr = '1' and p_in.spr_num(3 downto 0) = "0010" then |
|
mmcra <= p_in.spr_val; |
|
-- we don't support random sampling yet |
|
mmcra(MMCRA_SE) <= '0'; |
|
end if; |
|
if p_in.mtspr = '1' and p_in.spr_num(3 downto 0) = "1100" then |
|
siar <= p_in.spr_val; |
|
elsif doalert = '1' then |
|
siar <= p_in.nia; |
|
end if; |
|
if p_in.mtspr = '1' and p_in.spr_num(3 downto 0) = "1101" then |
|
sdar <= p_in.spr_val; |
|
elsif doalert = '1' then |
|
sdar <= p_in.addr; |
|
end if; |
|
if p_in.mtspr = '1' and p_in.spr_num(3 downto 0) = "0000" then |
|
sier <= p_in.spr_val; |
|
elsif doalert = '1' then |
|
sier <= (others => '0'); |
|
sier(SIER_SAMPPR) <= p_in.pr_msr; |
|
sier(SIER_SIARV) <= '1'; |
|
sier(SIER_SDARV) <= p_in.addr_v; |
|
end if; |
|
end if; |
|
prev_tb <= p_in.tbbits; |
|
end if; |
|
end process; |
|
|
|
pmu_2: process(all) |
|
variable tbdiff : std_ulogic_vector(3 downto 0); |
|
variable tbbit : std_ulogic; |
|
variable freeze : std_ulogic; |
|
variable event : std_ulogic; |
|
variable j : integer; |
|
variable inc : std_ulogic_vector(1 to 6); |
|
variable fc14wo : std_ulogic; |
|
begin |
|
event := '0'; |
|
|
|
-- Check for timebase events |
|
tbdiff := p_in.tbbits and not prev_tb; |
|
if is_X(mmcr0) then |
|
tbbit := 'X'; |
|
else |
|
tbbit := tbdiff(3 - to_integer(unsigned(mmcr0(MMCR0_TBSEL + 1 downto MMCR0_TBSEL)))); |
|
end if; |
|
if tbbit = '1' and mmcr0(MMCR0_TBEE) = '1' then |
|
event := '1'; |
|
end if; |
|
|
|
-- Check for counter negative events |
|
if mmcr0(MMCR0_PMC1CE) = '1' and pmcs(1)(31) = '1' then |
|
event := '1'; |
|
end if; |
|
if mmcr0(MMCR0_PMCjCE) = '1' and |
|
(pmcs(2)(31) or pmcs(3)(31) or pmcs(4)(31)) = '1' then |
|
event := '1'; |
|
end if; |
|
if mmcr0(MMCR0_PMCjCE) = '1' and |
|
mmcr0(MMCR0_PMCC + 1 downto MMCR0_PMCC) /= "11" and |
|
(pmcs(5)(31) or pmcs(6)(31)) = '1' then |
|
event := '1'; |
|
end if; |
|
|
|
-- Event selection |
|
inc := (others => '0'); |
|
fc14wo := '0'; |
|
case mmcr1(31 downto 24) is |
|
when x"f0" => |
|
inc(1) := '1'; |
|
fc14wo := '1'; -- override MMCR0[FC1_4WAIT] |
|
when x"f2" | x"fe" => |
|
inc(1) := p_in.occur.instr_complete; |
|
when x"f4" => |
|
inc(1) := p_in.occur.fp_complete; |
|
when x"f6" => |
|
inc(1) := p_in.occur.itlb_miss; |
|
when x"f8" => |
|
inc(1) := p_in.occur.no_instr_avail; |
|
when x"fa" => |
|
inc(1) := p_in.run; |
|
when x"fc" => |
|
inc(1) := p_in.occur.ld_complete; |
|
when others => |
|
end case; |
|
|
|
case mmcr1(23 downto 16) is |
|
when x"f0" => |
|
inc(2) := p_in.occur.st_complete; |
|
when x"f2" => |
|
inc(2) := p_in.occur.dispatch; |
|
when x"f4" => |
|
inc(2) := p_in.run; |
|
when x"f6" => |
|
inc(2) := p_in.occur.dtlb_miss_resolved; |
|
when x"f8" => |
|
inc(2) := p_in.occur.ext_interrupt; |
|
when x"fa" => |
|
inc(2) := p_in.occur.br_taken_complete; |
|
when x"fc" => |
|
inc(2) := p_in.occur.icache_miss; |
|
when x"fe" => |
|
inc(2) := p_in.occur.dc_miss_resolved; |
|
when others => |
|
end case; |
|
|
|
case mmcr1(15 downto 8) is |
|
when x"f0" => |
|
inc(3) := p_in.occur.dc_store_miss; |
|
when x"f2" => |
|
inc(3) := p_in.occur.dispatch; |
|
when x"f4" => |
|
inc(3) := p_in.occur.instr_complete and p_in.run; |
|
when x"f6" => |
|
inc(3) := p_in.occur.dc_ld_miss_resolved; |
|
when x"f8" => |
|
inc(3) := tbbit; |
|
when x"fe" => |
|
inc(3) := p_in.occur.dtlb_miss; |
|
when others => |
|
end case; |
|
|
|
case mmcr1(7 downto 0) is |
|
when x"f0" => |
|
inc(4) := p_in.occur.dc_load_miss; |
|
when x"f2" => |
|
inc(4) := p_in.occur.dispatch; |
|
when x"f4" => |
|
inc(4) := p_in.run; |
|
when x"f6" => |
|
inc(4) := p_in.occur.br_mispredict; |
|
when x"f8" => |
|
inc(4) := p_in.occur.ipref_discard; |
|
when x"fa" => |
|
inc(4) := p_in.occur.instr_complete and p_in.run; |
|
when x"fc" => |
|
inc(4) := p_in.occur.itlb_miss_resolved; |
|
when x"fe" => |
|
inc(4) := p_in.occur.ld_miss_nocache; |
|
when others => |
|
end case; |
|
|
|
inc(5) := (mmcr0(MMCR0_CC56RUN) or p_in.run) and p_in.occur.instr_complete; |
|
inc(6) := mmcr0(MMCR0_CC56RUN) or p_in.run; |
|
|
|
-- Evaluate freeze conditions |
|
freeze := mmcr0(MMCR0_FC) or |
|
(mmcr0(MMCR0_FCS) and not p_in.pr_msr) or |
|
(mmcr0(MMCR0_FCP) and not mmcr0(MMCR0_FCPC) and p_in.pr_msr) or |
|
(not mmcr0(MMCR0_FCP) and mmcr0(MMCR0_FCPC) and p_in.pr_msr) or |
|
(mmcr0(MMCR0_FCM1) and p_in.pmm_msr) or |
|
(mmcr0(MMCR0_FCM0) and not p_in.pmm_msr); |
|
|
|
if freeze = '1' or mmcr0(MMCR0_FC1_4) = '1' or |
|
(mmcr0(MMCR0_FC1_4W) = '1' and p_in.run = '0' and fc14wo = '0') then |
|
inc(1) := '0'; |
|
end if; |
|
if freeze = '1' or mmcr0(MMCR0_FC1_4) = '1' or |
|
(mmcr0(MMCR0_FC1_4W) = '1' and p_in.run = '0') then |
|
inc(2 to 4) := "000"; |
|
end if; |
|
if freeze = '1' or mmcr0(MMCR0_FC5_6) = '1' then |
|
inc(5 to 6) := "00"; |
|
end if; |
|
if mmcr0(MMCR0_TRIGGER) = '1' then |
|
inc(2 to 6) := "00000"; |
|
end if; |
|
for i in 1 to 6 loop |
|
j := (i - 1) * 9; |
|
if (mmcr2(MMCR2_FC0S - j) = '1' and p_in.pr_msr = '0') or |
|
(mmcr2(MMCR2_FC0P0 - j) = '1' and p_in.pr_msr = '1') or |
|
(mmcr2(MMCR2_FC0M1 - j) = '1' and p_in.pmm_msr = '1') or |
|
(mmcr2(MMCR2_FC0M1 - j) = '1' and p_in.pmm_msr = '1') then |
|
inc(i) := '0'; |
|
end if; |
|
end loop; |
|
|
|
-- When MMCR0[PMCC] = "11", PMC5 and PMC6 are not controlled by the |
|
-- MMCRs and don't generate events, but do continue to count run |
|
-- instructions and run cycles. |
|
if mmcr0(MMCR0_PMCC + 1 downto MMCR0_PMCC) = "11" then |
|
inc(5) := p_in.run and p_in.occur.instr_complete; |
|
inc(6) := p_in.run; |
|
end if; |
|
|
|
doinc <= inc; |
|
doevent <= event; |
|
doalert <= event and mmcr0(MMCR0_PMAE); |
|
end process; |
|
|
|
end architecture behaviour;
|
|
|