use work.gen_pkg.all;
entity alu is
- port
- (
+ port (
sys_clk : in std_logic;
sys_res_n : in std_logic;
opcode : in alu_ops;
end entity alu;
architecture beh of alu is
- type ALU_STATE is (SIDLE, SADD, SSUB, SMUL, SDIV, SDIV_CALC, SDONE, SERROR);
+ type ALU_STATE is (SIDLE, SADD, SSUB, SMUL, SDIV, SDIV_SHIFT_TO_MSB,
+ SDIV_CALC, SDIV_DONE, SDONE, SERROR);
signal state_int, state_next : ALU_STATE;
- signal done_intern, error_intern : std_logic;
- signal div_go_calc_int : std_logic;
signal op3_int, op3_next, opM_int, opM_next : csigned;
signal calc_done_int, calc_done_next : std_logic;
signal calc_error_int, calc_error_next : std_logic;
-- signale fuer division
- signal dividend_msb_int, dividend_msb_next, laengediv_int, laengediv_next : divinteger;
- signal quo_int, quo_next, aktdiv_int, aktdiv_int_next, op1_int, op1_next, op2_int, op2_next : csigned;
+ signal laengediv_int, laengediv_next : divinteger;
+ signal quo_int, quo_next, op1_int, op1_next, op2_int, op2_next : csigned;
signal sign_int, sign_next : std_logic;
begin
op3 <= op3_int;
calc_done_int <= '0';
calc_error_int <= '0';
--div
- dividend_msb_int <= (others => '0');
laengediv_int <= (others => '0');
quo_int <= (others => '0');
- aktdiv_int <= (others => '0');
op1_int <= (others => '0');
op2_int <= (others => '0');
sign_int <= '0';
calc_done_int <= calc_done_next;
calc_error_int <= calc_error_next;
-- div
- dividend_msb_int <= dividend_msb_next;
laengediv_int <= laengediv_next;
quo_int <= quo_next;
- aktdiv_int <= aktdiv_int_next;
op1_int <= op1_next;
op2_int <= op2_next;
sign_int <= sign_next;
end if;
end process;
- -- next state
- process(state_int, opcode, done_intern, error_intern, do_calc, div_go_calc_int)
+ -- next state & out
+ process(opcode, do_calc, state_int, op1, op2, laengediv_int, quo_int,
+ sign_int, op1_int, op2_int, op3_int, opM_int)
+ variable multmp : signed(((2*CBITS)-1) downto 0);
+ variable mulsign : std_logic;
+ variable tmp : csigned;
+ -- vars fuer div
+ variable laengediv_var, divtmp : divinteger;
+ variable aktdiv_int_var, op1_var, op2_var : csigned;
+ variable quobit : std_logic;
begin
+ calc_done_next <= '0';
+ calc_error_next <= '0';
+ -- default fuer div
+ laengediv_next <= laengediv_int;
+ quo_next <= quo_int;
+ op1_next <= op1_int;
+ op2_next <= op2_int;
+ sign_next <= sign_int;
+ op3_next <= op3_int;
+ opM_next <= opM_int;
+
-- set a default value for next state
state_next <= state_int;
-- next state berechnen
case state_int is
when SIDLE =>
+ sign_next <= '0';
+ op1_next <= (others => '0');
+ op2_next <= (others => '0');
+ opM_next <= (others => '0');
+
if do_calc = '1' then
case opcode is
when ALU_ADD => state_next <= SADD;
when others => state_next <= SIDLE;
end case;
end if;
- when SADD | SSUB | SMUL | SDIV | SDIV_CALC =>
- case state_int is
- when SDIV =>
- if div_go_calc_int = '1' then
- state_next <= SDIV_CALC;
- end if;
- when others => null;
- end case;
- if done_intern = '1' then
- state_next <= SDONE;
- end if;
- if error_intern = '1' then
- state_next <= SERROR;
- end if;
- when SDONE | SERROR =>
- if do_calc = '0' then
- state_next <= SIDLE;
- end if;
- end case;
- end process;
-
- -- output
- process(state_int, op1, op2, dividend_msb_int, laengediv_int, quo_int,
- aktdiv_int, sign_int, op1_int, op2_int, op3_int, opM_int)
- variable multmp, multmp2 : signed(((2*CBITS)-1) downto 0);
- variable mulsign : std_logic;
- variable tmp : csigned;
- -- vars fuer div
- variable laengediv_var, dividend_msb_var, divtmp : divinteger;
- variable aktdiv_int_var, quo_var, op1_var, op2_var : csigned;
- begin
- calc_done_next <= '0';
- calc_error_next <= '0';
- div_go_calc_int <= '0';
- done_intern <= '0';
- error_intern <= '0';
- -- default fuer div
- dividend_msb_next <= (others => '0');
- laengediv_next <= (others => '0');
- quo_next <= (others => '0');
- aktdiv_int_next <= aktdiv_int;
- op1_next <= (others => '0');
- op2_next <= (others => '0');
- sign_next <= '0';
- op3_next <= op3_int;
- opM_next <= opM_int;
-
- case state_int is
- when SIDLE =>
- opM_next <= (others => '0');
- aktdiv_int_next <= (others => '0');
when SADD =>
tmp := op1 + op2;
op3_next <= tmp;
+ state_next <= SDONE;
-- over- bzw. underflow?
if (op1(CBITS-1) = op2(CBITS-1)) and (op1(CBITS-1) /= tmp(CBITS -1)) then
- error_intern <= '1';
- else
- done_intern <= '1';
+ state_next <= SERROR;
end if;
when SSUB =>
tmp := op1 - op2;
op3_next <= tmp;
+ state_next <= SDONE;
-- over- bzw. underflow?
if (op1(CBITS-1) /= op2(CBITS-1)) and (op1(CBITS-1) /= tmp(CBITS -1)) then
- error_intern <= '1';
- else
- done_intern <= '1';
+ state_next <= SERROR;
end if;
when SMUL =>
mulsign := op1(CBITS-1) xor op2(CBITS-1);
op3_next(CBITS-1) <= mulsign;
if mulsign = '1' then
- multmp2 := (not multmp) + 1;
- else
- multmp2 := multmp;
+ multmp := (not multmp) + 1;
end if;
+ state_next <= SDONE;
-- overflow?
- if(multmp2((2*CBITS)-2 downto (CBITS-1)) > 0) then
- error_intern <= '1';
- else
- done_intern <= '1';
+ if(multmp((2*CBITS)-2 downto (CBITS-1)) > 0) then
+ state_next <= SERROR;
end if;
-
when SDIV =>
- -- division implementiert nach ~hwmod/doc/division.pdf
- if ((op1 = x"80000000" and op2 = to_signed(-1, CBITS)) or op2 = to_signed(0, CBITS)) then
- error_intern <= '1';
+ -- modifizierte binaere division nach ~hwmod/doc/division.pdf
+
+ -- division durch 0 checken
+ if op2 = to_signed(0, CBITS) then
+ state_next <= SERROR;
else
-- sign check
op1_var := op1;
op2_var := (not op2_var) + 1;
end if;
- dividend_msb_var := divinteger(find_msb(std_logic_vector(op1_var)))-1;
- laengediv_var := divinteger(find_msb(std_logic_vector(op2_var)))-1;
-
- aktdiv_int_next <= op1_var srl to_integer(dividend_msb_var - laengediv_var + 1);
-
+ state_next <= SDONE;
-- anmerkung: xst (xilinx) kann folgende zeile nicht uebersetzen
-- > if op1 = to_signed(-2147483648, CBITS) then
-- darum folgende schreibweise ->
if op1 = x"80000000" then
- -- so ziemlich das boeseste was passieren kann
- done_intern <= '1';
- op3_next <= to_signed(-214748364,CBITS);
- opM_next <= to_signed(8,CBITS);
+ if op2 = to_signed(10,CBITS) then
+ -- so ziemlich das boeseste was passieren kann
+ -- deswegen, weil divisor und dividend in positive
+ -- zahlen umgewandelt werden.
+ -- daher: hardcodiertes ergebnis fuer division durch 10 (fuer parser)
+ op3_next <= to_signed(-214748364,CBITS);
+ opM_next <= to_signed(8,CBITS);
+ else
+ -- sonst fehler..
+ state_next <= SERROR;
+ end if;
elsif (op1_var < op2_var) then
- done_intern <= '1';
+ -- => man braucht nix grossartiges rechnen
op3_next <= to_signed(0,CBITS);
opM_next <= op1_var;
else
- div_go_calc_int <= '1';
- dividend_msb_next <= dividend_msb_var;
- laengediv_next <= laengediv_var;
+ -- spannend! signale initialisieren fuer den naechsten schritt
+ state_next <= SDIV_SHIFT_TO_MSB;
+ laengediv_next <= (others => '0');
quo_next <= (others => '0');
op1_next <= op1_var;
op2_next <= op2_var;
sign_next <= op1(CBITS-1) xor op2(CBITS-1);
end if;
end if;
+ when SDIV_SHIFT_TO_MSB =>
+ -- shifte divisor so lange nach links, bis MSB = '1' ist
+ if op2_int(CBITS-1) = '1' then
+ state_next <= SDIV_CALC;
+ else
+ -- und mitzaehlen wie oft geshiftet wurde
+ laengediv_next <= laengediv_int + 1;
+ op2_next <= op2_int sll 1;
+ end if;
when SDIV_CALC =>
- divtmp := dividend_msb_int - laengediv_int + 1;
+ multmp := (others => '0');
+ multmp(CBITS downto 0) := signed('0' & std_logic_vector(op1_int)) - signed('0' & std_logic_vector(op2_int));
- if divtmp > 0 then
- aktdiv_int_var := aktdiv_int sll 1;
- aktdiv_int_var(0) := op1_int(to_integer(divtmp) - 1);
+ -- beim sign bit sieht man ob sich op2_int in op1_int ausging oder nicht
+ -- es muss daher negiert werden
+ quobit := not multmp(CBITS);
- quo_var := quo_int sll 1;
- if aktdiv_int_var >= op2_int then
- quo_var(0) := '1';
- aktdiv_int_var := aktdiv_int_var - op2_int;
- end if;
+ -- ergebnis dieser runde uebernehmen
+ quo_next <= quo_int(CBITS-2 downto 0) & quobit;
- quo_next <= quo_var;
- aktdiv_int_next <= aktdiv_int_var;
- dividend_msb_next <= dividend_msb_int;
- laengediv_next <= laengediv_int + 1;
- op1_next <= op1_int;
- op2_next <= op2_int;
- sign_next <= sign_int;
+ -- wenn es sich ausging, soll neuer dividend uebernommen werden
+ if quobit = '1' then
+ op1_next <= multmp(CBITS-1 downto 0);
+ end if;
+
+ laengediv_next <= laengediv_int - 1;
+ -- divisor nach rechts shiften
+ op2_next <= op2_int srl 1;
+
+ -- wenn laengediv_int in *dieser* runde =0 ist dann ist man
+ -- fertig, weil man braucht (runden zum shiften + 1) runden zum
+ -- berechnen des eregbnisses
+ if laengediv_int = 0 then
+ state_next <= SDIV_DONE;
+ end if;
+ when SDIV_DONE =>
+ if sign_int = '1' then
+ op3_next <= (not quo_int) + 1;
else
- if sign_int = '1' then
- op3_next <= (not quo_int) + 1;
- else
- op3_next <= quo_int;
- end if;
- opM_next <= aktdiv_int;
- done_intern <= '1';
+ op3_next <= quo_int;
end if;
+ -- wichtig: rest nur fuer postive paare gueltig!
+ -- (fuer unsere zwecke ausreichend)
+ opM_next <= op1_int;
+ state_next <= SDONE;
when SDONE | SERROR =>
calc_done_next <= '1';
- if (state_int = SERROR) then
+ if state_int = SERROR then
calc_error_next <= '1';
end if;
+ if do_calc = '0' then
+ state_next <= SIDLE;
+ end if;
end case;
end process;
end architecture beh;
-