giovafrankie
24-11-2015, 17:20
Sto realizzando una Floating Point Unit a precisione variabile e avrei bisogno di un modulo che faccia la normalizzazione.
L'idea è scorrere il mio vettore di bit e, non appena trovo il primo 1 (da sinistra), scrivere dentro una variabile (out_shift_esp_int) di quanto è lo shift (poi un altro modulo lo effettua, sfruttando quel numero).
Ho già realizzato un codice che lo fa in maniera "fissa":
--SI OCCUPA DI NORMALIZZARE IL NUMERO POST OPERAZIONE, CIOÈ SHIFTA IL RISULTATO A SINISTRA PER PORTARE IL PRIMO 1 NELLA POSIZIONE CORRETTA
encoder_priority: process(out_mux_normalizzatore, out_mux_esp)
variable out_shift_esp_appoggio: std_logic_vector(6 downto 0);
begin
--IMPLEMENTO UNA ENCODER PRIORITY, TROVANDO DIRETTAMENTE LO SHIFT
if out_mux_normalizzatore(23)='1'then out_shift_esp_int<="1111111"; -- lo shift è -1 111111
elsif out_mux_normalizzatore(22)='1' then out_shift_esp_int<="0000000"; -- lo shift è 0
elsif out_mux_normalizzatore(21)='1' then out_shift_esp_int<="0000001"; -- lo shift è 1 verso sx
elsif out_mux_normalizzatore(20)='1' then out_shift_esp_int<="0000010"; -- lo shift è 2 verso sx
elsif out_mux_normalizzatore(19)='1' then out_shift_esp_int<="0000011"; -- lo shift è 3 verso sx
elsif out_mux_normalizzatore(18)='1' then out_shift_esp_int<="0000100"; -- lo shift è 4 verso sx
elsif out_mux_normalizzatore(17)='1' then out_shift_esp_int<="0000101"; -- lo shift è 5 verso sx
elsif out_mux_normalizzatore(16)='1' then out_shift_esp_int<="0000110"; -- lo shift è 6 verso sx
elsif out_mux_normalizzatore(15)='1' then out_shift_esp_int<="0000111"; -- lo shift è 7 verso sx
elsif out_mux_normalizzatore(14)='1' then out_shift_esp_int<="0001000"; -- lo shift è 8 verso sx
elsif out_mux_normalizzatore(13)='1' then out_shift_esp_int<="0001001"; -- lo shift è 9 verso sx
elsif out_mux_normalizzatore(12)='1' then out_shift_esp_int<="0001010"; -- lo shift è 10 verso sx
elsif out_mux_normalizzatore(11)='1' then out_shift_esp_int<="0001011"; -- lo shift è 11 verso sx
elsif out_mux_normalizzatore(10)='1' then out_shift_esp_int<="0001100"; -- lo shift è 12 verso sx
elsif out_mux_normalizzatore(9)='1' then out_shift_esp_int<="0001101"; -- lo shift è 13 verso sx
elsif out_mux_normalizzatore(8)='1' then out_shift_esp_int<="0001110"; -- lo shift è 14 verso sx
elsif out_mux_normalizzatore(7)='1' then out_shift_esp_int<="0001111"; -- lo shift è 15 verso sx
elsif out_mux_normalizzatore(6)='1' then out_shift_esp_int<="0010000"; -- lo shift è 16 verso sx
elsif out_mux_normalizzatore(5)='1' then out_shift_esp_int<="0010001"; -- lo shift è 17 verso sx
elsif out_mux_normalizzatore(4)='1' then out_shift_esp_int<="0010010"; -- lo shift è 18 verso sx
elsif out_mux_normalizzatore(3)='1' then out_shift_esp_int<="0010011"; -- lo shift è 19 verso sx
elsif out_mux_normalizzatore(2)='1' then out_shift_esp_int<="0010100"; -- lo shift è 20 verso sx
elsif out_mux_normalizzatore(1)='1' then out_shift_esp_int<="0010101"; -- lo shift è 21 verso sx
elsif out_mux_normalizzatore(0)='1' then out_shift_esp_int<="0010110"; -- lo shift è 22 verso sx
else out_shift_esp_int <= out_mux_esp; -- NON CI SONO 0 QUINDI DAL MUX ARRIVA TUTTI 000...00 QUINDI L'USCITA È 0
end if;
end process encoder_priority;
Ho successivamente realizzato una versione parametrica. Però non sembra funzionare (l'uscita non è quella che dovrebbe essere). :muro:
--SI OCCUPA DI NORMALIZZARE IL NUMERO POST OPERAZIONE, CIOÈ SHIFTA IL RISULTATO A SINISTRA PER PORTARE IL PRIMO 1 NELLA POSIZIONE CORRETTA
encoder_priority: process(out_mux_normalizzatore, out_mux_esp)
variable counter: integer := ((2*(n_mant+2))-1);
variable i: integer;
begin
--IMPLEMENTO UNA ENCODER PRIORITY, TROVANDO DIRETTAMENTE LO SHIFT
if out_mux_normalizzatore(counter)='1' then out_shift_esp_int <= (others=>'1'); -- lo shift è -1 111111
else
normalizer: for i in (counter-1) to 0
LOOP
if out_mux_normalizzatore(i)='1' then
out_shift_esp_int <= conv_std_logic_vector(integer(counter-i-1),n_esp+2);
exit;
end if;
END LOOP normalizer;
if (i=0) then out_shift_esp_int <= out_mux_esp;
end if;
end if;
end process encoder_priority;
Volevo chiedervi se avete altre idee per poter risolvere questo problema. Quello che non posso realizzare è la prima soluzione parametrica, perchè è impossibile realizzare degli IF annidiati dentro ad un LOOP.
Non è necessaria la conoscenza del linguaggio VHDL, sarebbe utile anche un consiglio sull'algoritmo da utilizzare, poi ci penso io ad implementarlo, se possibile.
Vi ringrazio in anticipo. :help:
P.S.: Non essendoci la possibilità di lasciare indentato il codice, ho provveduto a colorarlo, il maniera da renderlo più leggibile.
L'idea è scorrere il mio vettore di bit e, non appena trovo il primo 1 (da sinistra), scrivere dentro una variabile (out_shift_esp_int) di quanto è lo shift (poi un altro modulo lo effettua, sfruttando quel numero).
Ho già realizzato un codice che lo fa in maniera "fissa":
--SI OCCUPA DI NORMALIZZARE IL NUMERO POST OPERAZIONE, CIOÈ SHIFTA IL RISULTATO A SINISTRA PER PORTARE IL PRIMO 1 NELLA POSIZIONE CORRETTA
encoder_priority: process(out_mux_normalizzatore, out_mux_esp)
variable out_shift_esp_appoggio: std_logic_vector(6 downto 0);
begin
--IMPLEMENTO UNA ENCODER PRIORITY, TROVANDO DIRETTAMENTE LO SHIFT
if out_mux_normalizzatore(23)='1'then out_shift_esp_int<="1111111"; -- lo shift è -1 111111
elsif out_mux_normalizzatore(22)='1' then out_shift_esp_int<="0000000"; -- lo shift è 0
elsif out_mux_normalizzatore(21)='1' then out_shift_esp_int<="0000001"; -- lo shift è 1 verso sx
elsif out_mux_normalizzatore(20)='1' then out_shift_esp_int<="0000010"; -- lo shift è 2 verso sx
elsif out_mux_normalizzatore(19)='1' then out_shift_esp_int<="0000011"; -- lo shift è 3 verso sx
elsif out_mux_normalizzatore(18)='1' then out_shift_esp_int<="0000100"; -- lo shift è 4 verso sx
elsif out_mux_normalizzatore(17)='1' then out_shift_esp_int<="0000101"; -- lo shift è 5 verso sx
elsif out_mux_normalizzatore(16)='1' then out_shift_esp_int<="0000110"; -- lo shift è 6 verso sx
elsif out_mux_normalizzatore(15)='1' then out_shift_esp_int<="0000111"; -- lo shift è 7 verso sx
elsif out_mux_normalizzatore(14)='1' then out_shift_esp_int<="0001000"; -- lo shift è 8 verso sx
elsif out_mux_normalizzatore(13)='1' then out_shift_esp_int<="0001001"; -- lo shift è 9 verso sx
elsif out_mux_normalizzatore(12)='1' then out_shift_esp_int<="0001010"; -- lo shift è 10 verso sx
elsif out_mux_normalizzatore(11)='1' then out_shift_esp_int<="0001011"; -- lo shift è 11 verso sx
elsif out_mux_normalizzatore(10)='1' then out_shift_esp_int<="0001100"; -- lo shift è 12 verso sx
elsif out_mux_normalizzatore(9)='1' then out_shift_esp_int<="0001101"; -- lo shift è 13 verso sx
elsif out_mux_normalizzatore(8)='1' then out_shift_esp_int<="0001110"; -- lo shift è 14 verso sx
elsif out_mux_normalizzatore(7)='1' then out_shift_esp_int<="0001111"; -- lo shift è 15 verso sx
elsif out_mux_normalizzatore(6)='1' then out_shift_esp_int<="0010000"; -- lo shift è 16 verso sx
elsif out_mux_normalizzatore(5)='1' then out_shift_esp_int<="0010001"; -- lo shift è 17 verso sx
elsif out_mux_normalizzatore(4)='1' then out_shift_esp_int<="0010010"; -- lo shift è 18 verso sx
elsif out_mux_normalizzatore(3)='1' then out_shift_esp_int<="0010011"; -- lo shift è 19 verso sx
elsif out_mux_normalizzatore(2)='1' then out_shift_esp_int<="0010100"; -- lo shift è 20 verso sx
elsif out_mux_normalizzatore(1)='1' then out_shift_esp_int<="0010101"; -- lo shift è 21 verso sx
elsif out_mux_normalizzatore(0)='1' then out_shift_esp_int<="0010110"; -- lo shift è 22 verso sx
else out_shift_esp_int <= out_mux_esp; -- NON CI SONO 0 QUINDI DAL MUX ARRIVA TUTTI 000...00 QUINDI L'USCITA È 0
end if;
end process encoder_priority;
Ho successivamente realizzato una versione parametrica. Però non sembra funzionare (l'uscita non è quella che dovrebbe essere). :muro:
--SI OCCUPA DI NORMALIZZARE IL NUMERO POST OPERAZIONE, CIOÈ SHIFTA IL RISULTATO A SINISTRA PER PORTARE IL PRIMO 1 NELLA POSIZIONE CORRETTA
encoder_priority: process(out_mux_normalizzatore, out_mux_esp)
variable counter: integer := ((2*(n_mant+2))-1);
variable i: integer;
begin
--IMPLEMENTO UNA ENCODER PRIORITY, TROVANDO DIRETTAMENTE LO SHIFT
if out_mux_normalizzatore(counter)='1' then out_shift_esp_int <= (others=>'1'); -- lo shift è -1 111111
else
normalizer: for i in (counter-1) to 0
LOOP
if out_mux_normalizzatore(i)='1' then
out_shift_esp_int <= conv_std_logic_vector(integer(counter-i-1),n_esp+2);
exit;
end if;
END LOOP normalizer;
if (i=0) then out_shift_esp_int <= out_mux_esp;
end if;
end if;
end process encoder_priority;
Volevo chiedervi se avete altre idee per poter risolvere questo problema. Quello che non posso realizzare è la prima soluzione parametrica, perchè è impossibile realizzare degli IF annidiati dentro ad un LOOP.
Non è necessaria la conoscenza del linguaggio VHDL, sarebbe utile anche un consiglio sull'algoritmo da utilizzare, poi ci penso io ad implementarlo, se possibile.
Vi ringrazio in anticipo. :help:
P.S.: Non essendoci la possibilità di lasciare indentato il codice, ho provveduto a colorarlo, il maniera da renderlo più leggibile.