Basic VHDL Constructs

advertisement
VHDL - INTRODUCTION
1.
2.
3.
4.
5.
6.
7.
8.
VHDL is not case sensitive
Identifier must start with a letter
All statements end with a semi-colon
Comments precede with (--)
“<= “ - signal assignment
“:=“ - variable assignment
Variables are like C- type variables.
Signals have a one time value set
associated to it at any time.
IF – THEN - ELSE
General syntax is:
if (condition_1) then
begin … statements_1 …
end;
elsif (condition_2) then
begin … statements_2 …
end;
else
begin …statements _i …end;
end if;
STRUCTURAL MODELING
1. Consider an example of a Dual 2 by 1
multiplexer using structural modeling.
2. Components required :
2- input AND gate
2- input OR gate
Inverter
ENTITY OF THE INVERTER
ENTITY inv IS
PORT(
i1: IN BIT;
o1: OUT BIT );
END inv;
ARCHITECTURE OF THE
INVERTER
ARCHITECTURE single_delay OF inv IS
BEGIN
o1 <= not (i1) after 5ns;
END single_delay;
DESIGN OF A SIMPLE 2 INPUT
AND GATE WITH A DELAY
ENTITY and2 IS
PORT(
i1, i2 :IN BIT;
o1: OUT BIT);
END and2;
ARCHITECTURE single_delay OF and2 IS
BEGIN
o1 <= (i1 AND i2) after 8ns;
END single_delay;
DESIGN OF A SIMPLE 2 INPUT
OR GATE WITH A DELAY
ENTITY or2 IS
PORT(
i1, i2 :IN BIT;
o1: OUT BIT);
END or2;
ARCHITECTURE single_delay OF or2 IS
BEGIN
o1 <= (i1 or i2) after 8ns;
END single_delay;
SINGLE AND DUAL 2-1
MULTIPLEXER ENTITY
ENTITY single_mux IS
PORT(
s: IN BIT ; --select input
iA, iB : INT BIT; -- iA is selected if
--s=‘0’ and iB is selected if s= ‘1’
oZ: OUT BIT –output of the mux
);
END single_mux;
SINGLE AND DUAL 2-1
MULTIPLEXER ARCHITECTURE
ARCHITECTURE gate_level OF single_mux IS
COMPONENT c1 PORT (
i1 :IN BIT;
o1: OUT BIT);
END COMPONENT;
COMPONENT c2 PORT (
i1, i2 :IN BIT;
o1: OUT BIT);
END COMPONENT;
SINGLE AND DUAL 2-1
MULTIPLEXER ARCHITECTURE
COMPONENT c3 PORT (
i1, i2 :IN BIT;
o1: OUT BIT);
END COMPONENT;
FOR ALL: c1 USE ENTITY WORK.inv(single_delay);
FOR ALL: c2 USE ENTITY WORK.and2(single_delay);
FOR ALL: c3 USE ENTITY WORK.or2(single_delay);
SIGNAL im1, im2, im3: BIT;
SINGLE AND DUAL 2-1
MULTIPLEXER ARCHITECTURE
BEGIN
g0: c1 PORT MAP (s, im1);
g1: c2 PORT MAP (iA, im1, im2);
g2: c2 PORT MAP (iB, s, im3);
g3: c3 PORT MAP (im2, im3, oZ);
END gate_level;
DUAL 2-1 MULTIPLEXER
ENTITY
ENTITY dual_mux IS
PORT(
s: IN BIT ; --select input
iA, iB : IN BIT _VECTOR(1
DOWNTO 0); -- iA is selected if -s=‘0’ and iB is selected if s= ‘1’
outZ: OUT BIT _VECTOR(1
DOWNTO 0)--output of the mux
);
END dual_mux;
DUAL 2-1 MULTIPLEXER
ARCHITECTURE
ARCHITECTURE element_level OF dual_mux IS
--we use only one component (the single 2:1 mux)
COMPONENT c1 PORT (s, iA, iB: IN BIT;
oZ : OUT BIT);
END COMPONENT;
FOR ALL : c1 USE ENTITY WORK. single_mux (gate_level);
BEGIN
g0 : c1 PORT MAP(sel, inA(0), inB(0), outZ(0)); --the 1st 2:1 mux is
--wired.
g1 : c1 PORT MAP(sel, inA(1), inB(1), outZ(1)); --the 2nd 2:! Mux is
--wired
END element_level;
BEHAVIORAL MODLEING
1. Show examples of a dual 2:1 multiplexer
designed using behavioral models.
2. Initially components built are:
2- input AND gate
2- OR gate
Inverter
DIFFERENT METHODS OF
BEHAVIORAL MODELING
1.
2.
3.
4.
5.
WHEN …ELSE statements
BOOLEAN operators
PROCESSES
WITH … SELECT
CASE statements
WHEN – ELSE STATEMENT
ENTITY dual_mux IS
PORT(
sel: IN BIT ; --select input
iA, iB : INT BIT_VECTOR( 1 DOWN TO 0); -- iA is selected
if --s=‘0’ and iB is selected if s= ‘1’
oZ: OUT BIT _VECTOR(1
DOWNTO 0)--output of the mux
);
END dual_mux;
ARCHITECTURE behaioral_when_else OF dual_mux IS
BEGIN
outZ <= inA WHEN(sel = ‘0’) ELSE inB;
END behavioral_when_else;
BOOLEAN OPERATORS
ARCHITECTURE behavioral_boolean OF
dual_mux IS
SIGNAL temp: BIT_VECTOR(1 downto 0);
BEGIN
temp <= (sel, sel);
--we assign sel to all the bits of the bus temp
outZ <= ((NOT) temp AND inA) OR (temp AND
inB);
END behavioral_boolean;
PROCESS
1. A process is a block that contains only
sequential statements.
2. Every process has a sensitivity list.
3. The signals in the sensitivity list determine
when the sequential statements within the
process will be executed.
PROCESS
ARCHITECTURE behavioral_process OF dual_mux IS
BEGIN
p1: PROCESS(sel, inA, inB) is
BEGIN
outZ <= inB;
if(sel = ‘0’) then
outZ <= inA;
End if;
END process p1;
END behavioral_process;
WITH ..SELECT STATEMENT
ARCHITECTURE behavioral_sel OF dual_mux
IS
BEGIN
WITH sel SELECT
outZ <=
inA after 10 ns when ‘0’,
inB after 10 ns when others;
END behavioral_sel;
CASE STATEMENT
ARCHITECTURE behavioral_sel OF dual_mux
IS
BEGIN
P1: PROCESS (sel, inA, inB) is
BEGIN
CASE sel is
when ‘0’ => outZ <= inA after 10 ns;
when others => outZ <= in B after 10 ns;
END CASE;
End process;
END behavioral_case;
TEST BENCHES
1. A VHDL block to test a design.
2. Example of a dual 2:1 multiplexer.
TEST BENCH
USE WORK. ALL;
ENTITY dual_mux_test IS
END dual_mux_test;
We do not need any interfacing since we only
want to use the test-bench with the
simulator.
TEST BENCH
ARCHITECTURE io_test OF dual_mux_test IS
COMPONENT c1 PORT(
sel: IN BIT;
inA, inB : IN BIT_VECTOR(1 downto 0);
outZ: OUT BIT_VECTOR(1 downto 0);
END COMPONENT;
SIGNAL sel_test: BIT;
SIGNAL inA_test, inB_test : BIT_VECTOR(1 downto 0);
SIGNAL outZ_test : BIT_VECTOR(1 downto 0);
TEST BENCH
BEGIN
g0: c1 PORT MAP (sel_test, inA_test,inB_test, outZ_test);
t1: inA_test <= “00”,
“01” after 50 ns,
“10” after 150 ns,
“11” after 250 ns,
“00” after 350 ns,
“01” after 450 ns,
“10” after 550 ns,
“11” after 650 ns;
t2: inB_test <= “00”,
“01” after 100 ns,
“10” after 200 ns,
“11” after 300 ns,
“00” after 400 ns,
“01” after 500 ns,
“10” after 600 ns,
“11” after 700 ns,
t3:sel_test <= ‘0’,
‘1’ after 325 ns;
END io_test;
GENERIC CLAUSE
GENERIC allows us to communicate nonhardware and non-signal information
between designs, and hence between
levels of hierarchy. In the example
discussed later, a delay is assigned a
default value of 5ns but the delay can be
changed when the component is used.
GENERIC CLAUSE
ENTITY decoder IS
GENERIC (
delay: TIME := 5ns );
PORT (
sel: IN BIT_VECTOR(2 downto 0);
outZ: OUT BIT_VECTOR(7 downto 0);
);
END decoder;
ARCHITECTURE single_delay OF decoder IS
BEGIN
WITH sel SELECT outZ <=
“00000001” after delay WHEN “000”,
“00000010” after delay WHEN “001”,
“00000100” after delay WHEN “010”,
“00001000” after delay WHEN “011”,
“00010000” after delay WHEN “100”,
“00100000” after delay WHEN “101”,
“01000000” after delay WHEN “110”,
“10000000” after delay WHEN “111”,
END single_delay;
GENERIC CLAUSE
ENTITY decoder_test IS
END decoder_test;
ARCHITECTURE io_test OF decoder_test IS
COMPONENT c1 GENERIC (delay: TIME);
PORT(
sel: IN BIT_VECTOR(2 downto 0);
outZ: OUT BIT_VECTOR( 7 downto 0));
END COMPONENT;
FOR g0: c1 USE ENTITY WORK. Decoder(single_delay);
SIGNAL sel_test : BIT_VECTOR(2 downto 0);
SIGNAL outZ_test : BIT_VECTOR(7 downto 0 );
BEGIN
g0 : c1 GENERIC MAP(17ns) PORT MAP(sel_test, outZ_test);
t1 : sel_test <= “000”
“001” after 50 ns,
“010” after 100 ns,
“011” after 150 ns,
“100” after 200 ns,
“101” after 250 ns,
“110” after 300 ns,
“111” after 350 ns,
“000” after 400 ns;
END io_test;
MEALY FINITE STATE
MACHINE(ASYNCHRONOUSLY)
MEALY FINITE STATE
MACHINE(ASYNCHRONOUSLY)
ENTITY asynchr_state_mach IS
PORT (
reset : IN BIT;
clk : IN BIT;
xin : IN BIT;
zout : OUT BIT;
stateout : OUT BIT_VECTOR(1 downto 0));
END asynchr_state_mach;
ARCHITECTURE behavioral OF asynchr_state_mach IS
TYPE state IS (state0, state1, state2, state3);
SIGNAL current_state, next_state : state := state0;
BEGIN
Cominat_proc : PROCESS (current_state, xin)
BEGIN
CASE current_state IS
MEALY FIFNITE STATE
MACHINE(ASYNCHRONOUSLY)
when state0 => stateout <= “00”;
if (xin = ‘0’) then
next_state <= state1; zout <= ‘1’;
else
next_state <= state0; zout <= ‘0’;
end if;
when state1 => stateout <= “01”;
If ( xin = ‘0’) then
next_state <= state1; zout <= ‘0’;
else
next_state <= state0; zout <= ‘0’;
end if;
When state2 => stateout <= “10”;
if (xin =‘0’) then
next_state <= state3; zout <= ‘1’;
else
next_state <= state0; zout <= ‘0’;
end if;
MEALY FIFNITE STATE
MACHINE(ASYNCHRONOUSLY)
When state3 => stateout <= “11”;
if (xin = ‘0’) then
next_state <= state0; zout <= ‘1’;
else
next_state <= state3; zout <= ‘1’;
end if;
end case;
end process combat_proc;
clk_proc : process
begin
wait until (clk’event and clk =‘1’);
if (reset = ‘1’) then
current_state <= state0;
else
current_state <= next_state;
end if;
end process clk_proc;
End behavioral;
MEALY FINITE STATE
MACHINE(ASYNCHRONOUSLY)
ENTITY USE WORK.ALL;
ENTITY asynchr_state_test IS
END asynchr_state_test;
ARCHITECTURE io_test OF synchr_state_test IS
BEGIN
COMPONENT c1 PORT (
reset : IN BIT;
clk : IN BIT;
xin : IN BIT;
Zout: IN BIT;
stateout: OUT BIT_VECTOR(1 downto 0);
END COMPONENT;
MEALY FIFNITE STATE
MACHINE(ASYNCHRONOUSLY)
FOR g0 : c1 USE ENTITY WORK. asynchr_state_mach(behavioral);
--we have to declare I/O signals that are used in --the test
SIGNAL reset_test : BIT:=‘0’;
SIGNAL clk_test : BIT:= ‘0’;
SIGNAL xin_test : BIT := ‘0’;
SIGNAL zout_test : BIT;
SIGNAL state_out_test: BIT_VECTOR( 1 downto 0);
BEGIN
clock_process : PROCESS is
BEGIN
clk_test <= NOT(clk_test);
wait for 20 ns;
END PROCESS;
g0 : c1 PORT MAP (reset_test, clk_test, xin_test, zout_test, stateout_test) ;
--apply a rest after 30 ns to go to state 0;
t1: reset_test <=
‘0’
‘1’ after 30ns,
‘0’ after 70 ns;
MEALY FIFNITE STATE
MACHINE(ASYNCHRONOUSLY)
t2 : xin_test <=
‘0’,
‘1’ after 50ns, --stay in state0
‘0’ after 90ns, --goto state1
‘1’ after 130ns, --stay in state1
‘0’ after 170ns, --goto state2
‘1’ after 210ns, --goto state3
‘0’ after 250ns, -stay in state3
‘1’ after 290ns, --goto state0
‘0’ after 330ns, --stay in state0
‘1’ after 370ns, --goto state1
‘0’ after 410ns, --goto state2
‘1’ after 450ns, --goto state3
‘0’ after 490ns; -- stay in state0
END io_test;
MOORE FINITE STATE MACHINE
MOORE FINITE STATE MACHINE
ENTITY(synchr_state_mach) IS
PORT(
reset : IN BIT;
clk: IN BIT;
xin : IN BIT;
zout : OUT BIT_VECTOR( 3 downto 0)
);
END synchr_state_mach;
MOORE FINITE STATE MACHINE
ARCHITECTURE behavioral OF synchr_state_mach IS
BEGIN
TYPE state IS (state0, state1, state2, state3);
SIGNAL current_state, next_state : state : state0;
BEGIN
combat_proc :PROCESS(current_state, xin) IS
BEGIN
case current_state is
when state0 =>
zout <= “0001”;
if ( xin = ‘0’) then
next_state <= state1;
else
next_state <= state0;
end if;
MOORE FINITE STATE MACHINE
when state1 =>
zout <= “0010”;
if ( xin = ‘0’) then
next_state <= state1;
else
next_state <= state2;
end if;
when state2 =>
zout <= “0100”;
if ( xin = ‘0’) then
next_state <= state3;
else
next_state <= state0;
end if;
when state3 =>
zout <= “1000”;
if ( xin = ‘0’) then
next_state <= state0;
else
next_state <= state3;
end if;
END CASE;
END PROCESS combinat_proc;
MOORE FINITE STATE MACHINE
Clk_proc : process(reset, clk) is
Begin
if (reset = ‘1’) then
current_state <= state0;
elsif (clk’event and clk = ‘1’) then
current_state <= next_state;
end if;
End process clk_proc;
End behavioral;
MOORE FINITE STATE MACHINE
USE WORK.ALL;
ENTITY synchr_state_mach_test IS
END synchr_state_mach_test;
ARCHITECTURE io_test OF synchr_state_mach_test
IS
COMPONENT c1 PORT (
reset : IN BIT;
clk : IN BIT;
xin : IN BIT;
Zout: IN BIT;
stateout: OUT BIT_VECTOR(1 downto 0);
END COMPONENT;
MOORE FINITE STATE MACHINE
FOR g0 : c1 USE ENTITY WORK. asynchr_state_mach(behavioral);
--we have to declare I/O signals that are used in --the test
SIGNAL reset_test : BIT:=‘0’;
SIGNAL clk_test : BIT:= ‘0’;
SIGNAL xin_test : BIT := ‘0’;
SIGNAL zout_test : BIT_VECTOR( 3 downto 0);
CONSTANT half_clk_period : TIME := 20ns;
CONSTANT clk_period : TIME := 40 ns;
BEGIN
clock_process : PROCESS
BEGIN
clk_test <= NOT( clk_test);
wait for half_clk_period;
END PROCESS clock_process;
g0: c1 PORT MAP (reset_test, clk_test, xin_test, zout_test);
t1: reset_test <= ‘0’,
‘1’ after (clk_period *0.75),
‘0’ after (clk_period *1.75);
MOORE FINITE STATE MACHINE
t2 : xin_test <= ‘0’, --start in state0
‘1’ after (clk_period *1.25), --stay in state0
‘0’ after (clk_period *2.25), --goto state1
‘0’ after (clk_period *3.25), --stay in state1
‘1’ after (clk_period *4.25), --goto state2
‘0’ after (clk_period *5.25), -- goto state3
‘1’ after (clk_period *6.25), --stay in state3
‘0’ after (clk_period *7.25), -- goto state0
‘1’ after (clk_period *8.25), --stay in state0
‘0’ after (clk_period *9.25), -- goto state1
‘1’ after (clk_period *10.25), -- goto state2
‘1’ after (clk_period *11.25), -- goto state0
‘1’ after (clk_period *12.25); --stay in state0
END io_test;
TWO PHASE CLOCK
• This entity generates 2 non-overlapping clocks of period 20ns.
ENTITY two_phase_clk IS
PORT(
phi1 : OUT BIT;
phi2 : OUT BIT );
END two_phase_clk;
ARCHITECTURE behave OF two_phase_clk IS
BEGIN
P1:process
begin
phi1 <= ‘0’, ‘1’ after 2ns, ‘0’ after 8ns;
phi2 <= ‘0’, ‘1’ after 12ns, ‘0’ after 18ns;
Wait for 20ns ; --period of 2 clocks=20ns
END PROCESS two _phase_clk;
END behave;
D FLIP-FLOP WITH
ASYNCHRONOUS RESET
This entity is a negative edge triggered D flipflop.
ENTITY asynchr_reset_dff IS
PORT(
clk: IN BIT;
reset: IN BIT;
din : IN BIT;
qout : OUT BIT;
qout_bar: OUT BIT);
END asynchr_reset_dff;
D FLIP-FLOP WITH
ASYNCHRONOUS RESET
ARCHITECTURE behave OF asynchr_rest_dff IS
Begin
p1: process(reset, clk) is
Begin
if( reset = ‘1’) then
qout <= ‘0’;
qout_bar <= ‘1’;
elsif (clk’event and clk = ‘0’) then
qout <= din;
qout_bar <= NOT(din);
end if;
end process p1;
end behave;
FOUR BIT COUNTER WITH
CLEAR AND ENABLE
PACKAGE own_types IS
TYPE bit4 IS RANGE 0 to 15;
TYPE four_state_logic IS (‘0’, ‘1’, ‘Z’, ‘X’);
END own_types;
--This entity is a 4 bit binary counter
USE WORK.own_types. ALL;
ENTITY counter_4bit IS
PORT(
clk : IN BIT;
enable : IN BIT;
clear : IN BIT;
count4b : INOUT BIT4 );
END counter_4bit;
FOUR BIT COUNTER WITH
CLEAR AND ENABLE
ARCHITECTURE behave OF counter_4bit IS
Signal counter_val : bit4;
Begin
p1 : process(enable, count4b) is
begin
if(enable = ‘0’) then
counter_val <= count4b;
elsif (count4b >= 15) then
counter_val <=0;
else
counter_val <= count4b + 1;
end if;
end process p1;
FOUR BIT COUNTER WITH
CLEAR AND ENABLE
P2 : process (clear, clk) is
Begin
if(clear = ‘1’) then
count4b <= 0;
elsif (clk_event and clk = ‘1’) then
count4b <= counter_val;
end if;
End process p2;
End behave;
FOUR BIT COUNTER WITH
CLEAR AND ENABLE
USE WORK. ALL
USE WORK.own_types.ALL;
Entity counter_4bit_test IS
End counter_4bit_test;
Architecture io_test OF counter_4bit_test IS
Component c1
PORT(
clk : IN BIT;
enable: IN BIT;
clear : IN BIT;
count4b : INOUT bit4;
);
End component;
FOUR BIT COUNTER WITH
CLEAR AND ENABLE
For g0: c1 USE ENTITY WORK.counter_4bit(behave);
Signal clk_test : BIT := ‘0’;
Signal enable_test : BIT := ‘0’;
Signal clear_test : BIT := ‘0’;
Signal count4b_test :bit4;
Begin
clk_process: process(clk_test) is
Begin
clk_test <= NOT(clk_test);
wait for 20ns;
End process clk_process;
FOUR BIT COUNTER WITH
CLEAR AND ENABLE
g0 : c1 port map (clk_test, enable_test, clear_test, count4b_test);
--we set enable to 0 when we want to check if
--the counter is correctly “frozen”.
enable_test <= ‘0’ after 50ns,
‘1’ after 100ns,
‘0’ after 210ns,
‘1’ after 310ns;
--we apply a clear after 30 ns
t1: clear_test <= ‘0’,
‘1’ after 30ns,
‘0’ after 70ns;
End io_test;
3 TYPES OF DELAYS
1.
•
2.
•
3.
•
•
Transport
Delay independent of input width
Inertial
Delay while reject small pulses
Delta
Minimum delay
Δ- delays do not accumulate
MODELING PROPOGATION
DELAY
•
•
Signal must remain constant for at least the
propagation delay or greater than the reject
time to be seen on the output
Pulse rejection can be less than
propagation - must be explicitly stated and
must be less that inertial delay, e.g.,
Out <= reject 6ns inertial ( A or B)
after 8ns;
MODELING PROPOGATION
DELAY
•
•
To specify a delay between the input and
the output of a conditional concurrent
statement, append :
after time
Inertial (Propagation) delay e.g.,
S_out <= (S xor B xor C_in) after 10ns;
INERTIAL DELAY
•
•
Provides for specification of input pulse width, i.e.
‘inertia’ of output, and propagation delay
Inertial delay is default and reject is optional
TRANSPORT DELAY
•
•
Delay must be explicitly specified by user
• keyword transport must be used.
Signal will assume its new value after specified
delay, independent of width.
INERTIAL DELAY, e.g.,
•
Inverter with propagation delay of 10ns
which suppresses pulses shorter than 5ns.
output <= reject 5ns inertial not input after
10ns;
NO DELTA DELAY, e.g.,
DELTA DELAY, e.g.,
FIFO CODE
Library ieee;
Use ieee.std_logic_1164.all;
Use work.std_arith.all;
entity FIFO_LOGIC is
generic (N:integer := 3);
port(clk, push, pop, init : in std_logic;
add : out std_logic_vector(N-1 downto 0);
full,empty,we,nppush,nopop : buffer std_logic);
end FIFO_LOGIC;
architecture RTL of FIFO_LOGIC is
signal wptr, rptr: std_logic_vector(N-1 downto 0);
signal lastop :std_logic;
FIFO CODE
Begin
Sync: process(clk) is
Begin
if (clk’event and clk =‘1’) then
if (init = ‘1’) then
wptr <=(others => ‘0’);
rptr <= (others => ‘0’);
lastop <= ‘0’;
elsif (pop = ‘1’ and empty = ‘0’) then
rptr <= rptr +1;
lastop <= ‘0’;
elsif (push = ‘1’ and full = ‘0’) then
wptr <= wptr +1;
lastop <= ‘1’;
end if;
end if;
End process sync;
FIFO CODE
Comb: process ( push, pop, wptr, rptr, lastop, full, empty) is
begin
--full and empty flags—
if(rptr = wptr) then
if(lastop = ‘1’) then
full <= ‘1’;
empty <= ‘0’;
else
full <= ‘0’;
empty <= ‘1’;
end if;
else
full <= ‘0’;
empty <= ‘0’;
end if;
FIFO CODE
--address, write enable and no push/ no pop logic –
if(pop = ‘0’ and push = ‘0’) then -- no operation --add <= rptr;
we <= ‘0’;
nopush <= ‘0’;
nopop <= ‘0’;
elsif (pop = ‘0’ and push = ‘1’) then --push only-add <= wptr;
nopop <= ‘0’;
if(full = ‘0’) then
we <= ‘1’;
nopush <= ‘0’;
else
we <= ‘0’;
nopush <= ‘1’;
end if;
FIFO CODE
Elsif (pop = ‘1’ and push = ‘0’) then -- pop only -add <= rptr;
nopush <= ‘0’;
we <= ‘0’;
if(empty = ‘0’) then --valid read condition –
nopop <= ‘0’;
else
nopop <= ‘1’;
end if;
else
add <= rptr;
we <= ‘0’;
nopush <= ‘1’;
nopop <= ‘0’;
end if;
end process comb;
end architecture RTL;
ARITHMETIC LOGIC UNIT
Library ieee;
Use ieee.std_logic_1164.all;
Use ieee.std_lgic_misc.all;
Use ieee.std_logic_arith.all;
Use ieee.std_logic_unsigned.all;
Entity ALU is
generic(N:integer :=4);
port(clk, shift_dir, sl, sr : in std_logic;
Ain, Bin, memdata: in std_logic_vector(N-1 downto 0);
memack: in std_logic;
opcode: in std_logic_vector( 3 downto 0);
ACC: outstd_logic_vector(N-1 downto 0);
flag: out std_logic := ‘Z’);
end ALU;
ARITHMETIC LOGIC UNIT
Architecture BEHAVE of ALU is
Signal sout, a, b : std_logic_vector(N-1 downto 0);
Begin
func: process(clk, opcode, memack) is
Begin
case opcode is
when “0001” => sout <= a+b;
when “0011” => sout <= a-b;
when “0010” => sout <= a and b;
when “0100” => sout <= a or b;
when “0101” => sout <= not b;
when”0101” =>
if (clk’event and clk =‘1’) then
if( shift_dir = ‘0’) then
sout <= sout(N-2 downto 0) & s1;
else
sout <= sr & sout(N-1 downto 1);
end if;
end if;
ARITHMETIC LOGIC UNIT
when “0111” => sout <= b;
when “1001” => sout <= null;
when “1000” =>
if(memack = ‘1’) then
sout <= memdata;
end if;
when “1111” =>
if (a<=b) then
flag <= ‘1’;
else
flag <= ‘0’;
end if;
when “1110”=> null;
ARITHMETIC LOGIC UNIT
when “1100” => sout <= b;
when “0000” => null;
when others => null;
end case;
end process;
sync: process (clk) is
begin
if(clk,event and clk = ‘1’) then
a <= Ain;
b <= Bin;
ACC <= sout;
end if;
end process sync;
end BEHAVE;
ARITHMETIC LOGIC UNIT –
TEST BENCH
Library ieee;
Use ieee.std_logic_1164.all;
entity TB_ALU is
end TB_ALU;
architecture TEST_BENCH of T B_ALU is
begin
constant clk_hp : TIME := 50ns;
constant N : integer := 4;
signal clk, shift_dir, sl, sr : std_logic := ‘0’;
signal memack, flag : syd_logic := ‘0’;
signal Ain, Bin, memdata, ACC : std_logic_vector( N-1 downto 0);
signal opcode : std_logic_vector (3 downto 0);
ARITHMETIC LOGIC UNIT –
TEST BENCH
component ALU
generic (N: integer := 4);
port (clk, shift_dir, sl, sr : in std_logic;
Ain, Bin, memdata : in std_logic_vector(N-1 downto 0);
opcode : in std_logic_vector( 3 downto 0);
ACC : in std_logic_vector(N-1 downto 0);
flag : out std_logic : ‘Z’ );
end component;
begin
UUT : ALU
Port map (clk, shift_dir, sr, sl, Ain, Bin, memdata, memack, opcode,
ACC, flag);
ARITHMETIC LOGIC UNIT –
TEST BENCH
clockgen : process is
begin
clk <= ‘1’ after clk_hp, ‘0’ after 2*clk_hp;
wait for 2*clk_hp;
end process clockgen;
SignalSource : process is
begin
opcode <= “0111”, “0101”, after 30ns;
Bin <= “1111”, “0000” after 400ns;
wait;
end process SignalSource;
end TESTBENCH;
configuration CFG_TB_ALU of TB_ALU
is
for TESTBENCH
for UUT : ALU
end for;
end;
UP/DOWN COUNTER:
BEHAVIORAL
entity COUNTER is
PORT(
d: in integer range 0 to 255;
clk, clear, load, up_down : in bit;
qd : out integer range 0 to 255 );
end COUNTER;
architecture A of COUNTER is
begin
process (clk) is
variable cnt : integer range 0 to 255;
variable direction : integer;
begin
if (up_down = ‘1’) then
direction := 1;
else
direction := -1;
end if;
UP/DOWN COUNTER:
BEHAVIORAL
if (clk’event and clk = ‘1’) then
if ( load := 1) then
cnt := d;
else
cnt := cnt + direction;
end if;
-- the following lines will produce a synchronous clear on the
--counter
if( clear = ‘0’) then
cnt := ‘0’;
end if;
end if;
qd <= cnt;
end process;
end A;
T FLIP-FLOP : STRUCTURAL
library ieee;
use ieee.std_logic_1164.all;
entity andgate is
port (A, B, C, D : in std_logic := ‘1’;
Y : out std_ulogic);
end andgate;
architecture gate of andgate is
begin
Y <= A and B and C and D;
end gate;
T FLIP-FLOP : STRUCTURAL
library ieee;
use ieee.std_logic_1164.all;
entity tff is
port (Rst, clk, T : in ustd_logic := ‘1’;
Q : out std_ulogic);
end tff;
architecture behave of tff is
begin
process (Rst, clk) is
variable qtmp :std_ulogic;
begin
if (Rst = ‘1’) then
qtmp := ‘0’;
elsif rising_edge( clk) then
if T = ‘1’ then
qtmp := not qtmp;
end if;
end if;
Q <= qtmp;
end process;
end behave;
T FLIP-FLOP : STRUCTURAL
library ieee;
use ieee.std_logic_1164.all;
entity TCOUNT is
port (Rst, clk : std_ulogic;
count : std_ulogic_vector ( 4 downto 0);
end TCOUNT;
architecture STRUCTURE of TCOUNT is
component tff
port ( Rst, clk, T : in std_ulogic;
Q : out std_ulogic );
end component;
component andgate
port ( A, B, C, D : in std_ulogic;
Y : out std_ulogic );
end component;
constant VCC : std_logic := ‘1’;
signal T, Q : std_ulogic_vector ( 4 downto 0);
T FLIP-FLOP : STRUCTURAL
begin
T(0) <= VCC;
T0 : tff port map (Rst => Rst, clk => clk, T => T(0), Q => Q(0) );
T(1) <= Q(0);
T1 : tff port map (Rst => Rst, clk => clk, T => T(1), Q => Q(1) );
A1 : andgate port map (A => Q(0), B => Q(1), Y => T(2) );
T2 : tff port map (Rst => Rst, clk => clk, T => T(2), Q => Q(2) );
A2 : andgate port map (A => Q(0), B => Q(1), C => Q(2), Y => T(2)
);
T3 : tff port map (Rst => Rst, clk => clk, T => T(3), Q => Q(3) );
A3 : andgate port map (A => Q(0), B => Q(1), C => Q(2), D =>
Q(3), Y => T(2) );
T4 : tff port map (Rst => Rst, clk => clk, T => T(4), Q => Q(4) );
count <= Q;
end STRUCTURE;
TRI-STATE BUS : DATAFLOW
library ieee;
use ieee.std_logic_1164.all;
entity prebus is
port ( my_in : in std_logic_vector ( 7 downto 0 );
sel : in std_logic;
my_out : out std_logic_vector (7 downto 0 );
end prebus;
architecture maxpid of prebus is
begin
my_out <= “ZZZZZZZZ”;
when (sel = ‘1’ )
else my_in;
end maxpid;
BARREL SHIFTER :
BEHAVIORAL
library ieee;
use ieee.std_logic_1164.all;
entity shifter is
port (clk, rst, load : in std_ulogic;
data : in std_ulogic( 0 to 7);
q : out std_ulogic ( 0 to 7) );
end shifter;
architecture behavior of shifter is
begin
process (rst, clk ) is
variable qreg : std_ulogic_vector ( 0 to 7);
begin
if rst = ‘1’ then
qreg := “00000000”;
BARREL SHIFTER :
BEHAVIORAL
elsif rising_edge(clk) then
if load = ‘1’ then
qreg := data;
else
qreg := qreg ( 1 to 7) & qreg(0); -- left shift
qreg := qreg (7) & qreg (0 to 6); -- right shift
end if;
end if;
q <= qreg;
end process;
end behavioral;
MODULO 6 COUNTER
library ieee;
use ieee.std_logic_1164.all;
use ieee.std _logic_unsigned.all;
entity count5 is
port ( clk, reset, enable : in std_logic;
count : out std_logiv_vector(3 downto 0) );
end count5;
architecture rtl of cout5 is
begin
cp : process (clk) is
variable qnext : std_logic_vector ( 3 downto 0);
begin
if reset = ‘1’ then
qnext := “0000”
elsif (enable = ‘1’) then
if (count <5 ) then
qnext := qnext +1;
else
qnext := “0000”
MODULO 6 COUNTER
end if;
else
qnext := count;
end if;
if (clk’event and clk = ‘1’) then
count <= qnext;
end if;
end process cp;
end rtl;
ONE HOT FSM
library ieee;
use ieee.std_logic_1164.all;
use ieee.std _logic_unsigned.all;
use ieee.std _logic_arith.all;
entity fum is
port (clk, reset, x, y, z : in std_logic;
f : out std_logic);
end fum;
architecture rtl of fum is
signal state_a : std_logic;
signal state_b : std_logic;
signal state_c : std_logic;
signal state_d : std_logic;
signal next_state_a : std_logic;
signal next_state_b : std_logic;
signal next_state_c : std_logic;
signal next_state_d : std_logic;
begin
ONE HOT FSM
output : process ( state_a, state_b, state_c, state_d) is
variable f_v : std_logic;
begin
f_v := ‘0’;
if (state_a = ‘1’) then
f_v := ‘1’;
end if;
if (state_b = ‘1’) then
f_v := ‘1’;
end if;
f <= f_v;
end process;
fsm : process (x, y, z, state_a, state_b, state_c, state_d ) is
variable vnext_state_a : std_logic;
variable vnext_state_b : std_logic;
variable vnext_state_c : std_logic;
variable vnext_state_d : std_logic;
begin
vnext_state_a := ‘0’;
ONE HOT FSM
vnext_state_b := ‘0’;
vnext_state_c := ‘0’;
vnext_state_d := ‘0’;
if (state_a = ‘1’) then
if ( x = ‘1’) then
vnext_state_c := ‘1’;
elsif (z = ‘1’) then
vnext_state_d := ‘1’;
else
vnext_state_b := ‘1’;
end if;
end if;
if (state_b = ‘1’) then
vnext_state_d := ‘1’;
end if;
ONE HOT FSM
if (state_b = ‘1’) then
if ( z = ‘1’) then
if ( x = ‘1’) then
vnext_state_c := ‘1’;
else
vnext_state_b := ‘1’;
end if;
else
vnext_state_d := ‘1’;
end if;
end if;
if (state_d = ‘1’) then
if ( x= ‘1’) then
vnext_state_b := ‘1’;
elsif ( y= ‘1’) then
vnext_state_c := ‘1’;
elsif ( z= ‘1’) then
vnext_state_a := ‘1’;
else
vnext_state_d := ‘1’;
end if;
end if;
ONE HOT FSM
next_state_a <= vnext_state_a;
next_state_b <= vnext_state_b;
next_state_c <= vnext_state_c;
next_state_d <= vnext_state_d;
end process fum;
reg_process : process (clk, reset, next_state_a, next_state_b, next_state_c, next_state_d ) is
begin
if (clk’event and clk = ‘1’ ) then
if (reset = ‘1’) then
state_a <= ‘1’;
state_b <= ‘0’;
state_c <= ‘0’;
state_d <= ‘0’;
else
state_a <= next_state_a;
state_b <= next_state_b;
state_c <= next_state_c;
state_d <= next_state_d;
end if;
end if;
end process reg_process;
end rtl;
A PIPELINE
library ieee;
use ieee.std_logic_1164.all;
use ieee.std _logic_unsigned.all;
use ieee.std _logic_arith.all;
entity pipeline is
port ( a : in std_logic_vector (6 downto 0);
clk, reset : in std_logic;
f : out std_logic_vector ( 6 downto 0) );
end pipeline;
architecture pipelined of pipeline is
signal b, c, d, e : std_logic_vector ( 6 downto 0);
begin
reg1 : process (clk, reset ) is
begin
if reset = ‘1’ then
b <= “0000000”;
elsif rising_edge (clk) then
b <= a;
end if;
end process reg1;
A PIPELINE
reg2 : process (clk, reset ) is
begin
if reset = ‘1’ then
c <= “0000000”;
elsif rising_edge (clk) then
c <= b + “0000001”;
end if;
end process reg2;
reg3 : process (clk, reset ) is
begin
if reset = ‘1’ then
d <= “0000000”;
elsif rising_edge (clk) then
d <= c + “0000010”;
end if;
end process reg3;
A PIPELINE
reg4 : process (clk, reset ) is
begin
if reset = ‘1’ then
e <= “0000000”;
elsif rising_edge (clk) then
e <= d + “0000011”;
end if;
end process reg4;
reg5 : process (clk, reset ) is
begin
if reset = ‘1’ then
f <= “0000000”;
elsif rising_edge (clk) then
f <= e + “0000100”;
end if;
end process reg5;
end process pipelined;
A GENERIC ADDER
library ieee;
use ieee.std_logic_1164.all;
use ieee.std _logic_unsigned.all;
use ieee.std _logic_arith.all;
entity add_g is
generic (left : natural := 31; -- top bit
prop : time :=100 ps );
port ( a, b : in std_logic_vector ( left downto 0 );
cin : in std_logic_vector;
sum : out std_logic_vector (left downto 0);
cout : out std_logic );
end entity add_g;
architecture behave of add_g is
begin
adder : process
variable carry : std_logic; -- internal signal
variable isum : std_logic_vector ( left downto 0);
A GENERIC ADDER
begin
carry := cin;
for I in 0 to left loop
isum(i) := a(i) xor b(i) xor carry;
carry := (a(i) and b (i) ) or (a(i) and carry ) or (b(i) and carry);
end loop;
sum <= isum;
cout <= carry;
wait for prop; -- signals updated after prop delay
end process adder;
end architecture behave;
16 : 1 MULTIPLEXER
library ieee;
use ieee.std_logic_1164.all;
entity mux4 is
port ( s: in std_logic_vector ( 1 downto 0 );
i : in std_logic_vector ( 3 downto 0 );
y : out std_logic );
end mux4;
architecture vector of mux4 is
begin
with s select
y <= i (0) when “00”,
i (1) when “01”,
i (2) when “10”,
i (3) when others;
end vector;
16 : 1 MULTIPLEXER
library ieee;
use ieee.std_logic_1164.all;
entity mux16 is
port ( datain: in std_logic_vector ( 15 downto 0 );
sel : in std_logic_vector ( 3 downto 0 );
dataout : out std_logic );
end mux16;
architecture rtl of mux16 is
component mux4
port ( s: in std_logic_vector ( 1 downto 0 );
i : in std_logic_vector ( 3 downto 0 );
y : out std_logic );
end component;
for all : mux4 use entity work. mux4(vector);
signal level1 : std_logic_vector (3 downto 0); -- intermediate signals
begin
16 : 1 MULTIPLEXER
mux0 :mux4
port map (s => sel (1 downto 0), i => datain(3 downto 0), y =>
level1 (0) );
mux1 :mux4
port map (s => sel (1 downto 0), i => datain(7 downto 4), y =>
level1 (1) );
mux2 :mux4
port map (s => sel (1 downto 0), i => datain(11 downto 8), y
=> level1 (2) );
mux3 :mux4
port map (s => sel (1 downto 0), i => datain(15 downto 12), y
=> level1 (3) );
muxall :mux4
port map (s => sel (1 downto 0), i => level1, y => dataout );
end rtl;
Download