2 use ieee.std_logic_1164.all;
3 use ieee.numeric_std.all;
9 sys_clk : in std_logic;
10 sys_res_n : in std_logic;
15 do_calc : in std_logic;
16 calc_done : out std_logic
20 architecture beh of alu is
21 type ALU_STATE is (SIDLE, SADD, SSUB, SMUL, SDIV, SDIV_CALC, SDIV_DONE, SDONE);
22 signal state, state_next : ALU_STATE;
23 signal done_intern, div_calc_done, div_go_calc : std_logic;
24 signal op3_int, op3_next : csigned;
25 signal calc_done_int, calc_done_next : std_logic;
26 -- signale fuer division
27 signal dividend_msb, dividend_msb_next, laengediv, laengediv_next : natural;
28 signal quo, quo_next, aktdiv, aktdiv_next, op1_int, op1_next, op2_int, op2_next : csigned;
29 signal sign, sign_next : std_logic;
32 calc_done <= calc_done_int;
35 process(sys_clk, sys_res_n)
37 if sys_res_n = '0' then
39 op3_int <= (others => '0');
44 quo <= (others => '0');
45 aktdiv <= (others => '0');
46 op1_int <= (others => '0');
47 op2_int <= (others => '0');
49 elsif rising_edge(sys_clk) then
52 calc_done_int <= calc_done_next;
54 dividend_msb <= dividend_msb_next;
55 laengediv <= laengediv_next;
57 aktdiv <= aktdiv_next;
65 process(state, opcode, done_intern, do_calc, div_calc_done, div_go_calc)
67 -- set a default value for next state
69 -- next state berechnen
87 if done_intern = '1' then
91 if done_intern = '1' then
95 if done_intern = '1' then
99 if div_go_calc = '1' then
100 state_next <= SDIV_CALC;
103 if div_calc_done = '1' then
104 state_next <= SDIV_DONE;
107 if done_intern = '1' then
111 if do_calc = '0' then
118 process(state, op1, op2, dividend_msb, laengediv, quo, aktdiv, sign, op1_int, op2_int)
119 variable tmperg : csigned;
120 variable multmp : signed(((2*CBITS)-1) downto 0);
122 variable laengediv_var, dividend_msb_var : natural;
123 variable aktdiv_var, quo_var, op1_var, op2_var : csigned;
125 op3_next <= (others => '0');
126 calc_done_next <= '0';
127 div_calc_done <= '0';
131 dividend_msb_next <= 0;
133 quo_next <= (others => '0');
134 aktdiv_next <= (others => '0');
135 op1_next <= (others => '0');
136 op2_next <= (others => '0');
141 tmperg := (others => '0');
150 tmperg(CBITS-1) := multmp((2*CBITS)-1);
151 tmperg((CBITS-2) downto 0) := multmp((CBITS-2) downto 0);
154 -- division implementiert nach ~hwmod/doc/division.pdf
155 tmperg := (others => '0');
156 if op2 = to_signed(0,CBITS) then
157 -- TODO: err out signal
163 if op1(CBITS-1) = '1' then
164 op1_var := not (op1_var + 1);
166 if op2(CBITS-1) = '1' then
167 op2_var := not (op2_var + 1);
170 dividend_msb_var := find_msb(op1_var)-1;
171 laengediv_var := find_msb(op2_var)-1;
173 aktdiv_next <= op1_var srl (dividend_msb_var - laengediv_var + 1);
176 dividend_msb_next <= dividend_msb_var;
177 laengediv_next <= laengediv_var;
178 quo_next <= (others => '0');
181 sign_next <= op1(CBITS-1) xor op2(CBITS-1);
184 tmperg := (others => '0');
186 if (dividend_msb - laengediv + 1) > 0 then
187 aktdiv_var := aktdiv sll 1;
188 aktdiv_var(0) := op1_int(dividend_msb - laengediv);
190 quo_var := quo sll 1;
191 if aktdiv_var >= op2_int then
193 aktdiv_var := aktdiv_var - op2_int;
197 aktdiv_next <= aktdiv_var;
198 dividend_msb_next <= dividend_msb;
199 laengediv_next <= laengediv + 1;
205 quo_next <= (not quo) + 1;
209 div_calc_done <= '1';
216 calc_done_next <= '1';
218 tmperg := (others => '0');
221 end architecture beh;