aspect
VHDL combinational circuits modeling - examples
prepared by P. Bakowski
Contents: simple adder, ones counter, latch with I/O logic
First example
A simple combinational circuit - full adder
In this section we will look at a small example of a VHDL description of a full adder to give you a feel for the language and how it is used.
We start the description of an entity by specifying its external interface, which includes a description of its ports.
Below we present three kinds of architectures for the full adder entity:
purely functional architecture
behavioral architecture including function and timing
tabular desription of the behavioral description
The first model expresses only one aspect of the full adder behavior which is function (function only)
entity adder is
port (a,b,cin: in bit; sum,cout: out bit);
end adder;
architecture rtl of adder is
begin
sum <= (a xor b) xor cin;
cout <= (a and b) or (cin and a) or (cin and b);
end rtl;
It may be developed in order to introduce timing information through the generic statement which contains the predefined temporal parameters - delays:
generic(sum_del:TIME:=10 ns; carry_del: TIME:= 15 ns);
Which provides us with a reformulated architecture which provides a complete behaviour including function and timing
sum <= (a xor b) xor cin after sum_del;
cout <= (a and b) or (cin and a) or (cin and b) after carry_del;
The following adder architecture is decribed using a table. This approach to built very rapid models requiring no calculations.
The tabular approach is extensively used to model simple combinational logic circuits.
architecture tabular of adder is
type entry is record
a,b,cin: bit; -- input values
cout,sum: bit; -- output values
end record;
type table is array (0 to 7) of entry;
constant truth_table: table :=
(-- a -- b -- cin ---- cout -- sum --
( '0' '0' '0' '0' '0'),
( '0' '0' '1' '0' '1'),
( '0' '1' '0' '0' '1'),
( '0' '1' '1' '1' '0'),
( '1' '0' '0' '0' '1'),
( '1' '0' '1' '1' '0'),
( '1' '1' '0' '1' '0'),
( '1' '1' '1' '1' '1'));
process(a,b,cin)
cout <= truth_table(bin2int(a&b&cin)).cout after carry_del;
sum<= truth_table(bin2int(a&b&cin)).sum after sum_del;
end process;
end tabular;
Note the use of bin2int conversion function. This function transforms a bit vector into the integer value necessary to use to select an element into array.
Second example
Ones counter
The second example describes a simple one's counter with three inputs and two outputs.
The model uses one VHDL process.
entity cmpt_1 is
port(a:bit_vector(0 to 2);c:bit_vector(0 to 1));
end cmpt_1;
architecture comport of cmpt_1 is
signal a:bit_vector(0 to 2);
signal c:bit_vector(0 to 1);
compt:process(a)
variable val:integer range 0 to 3;
val:=0;
for I in 0 to 2 loop
if a(I) = '1' then
val:= val+ 1;
end if;
end loop;
case val is
when 0 => c<= "00" after 10ns;
when 1 => c<= "01" after 10ns;
when 2 => c<= "10" after 10ns;
when 3 => c<= "11" after 10ns;
end case;
end process compt;
-- Test process
test:process
A <= "000" after 10ns, "001" after 20ns, "010" after 30ns,
"011" after 40ns, "100" after 50ns, "101" after 60ns,
"110" after 70ns, "111" after 80ns, "000" after 90ns;
wait;
end process test;
end comport;
Third example
Latch models
The behaviour of this circuit may be modeled with different degrees of precision depending on the temporal aspect.
The following example illustrate three levels of timing modeling for 8-bit latch.
no explicit timing or delta delay timing
simple I/O timing or simple one path delay timing
complete timing or multipath delay timing
The first model has a simplified timing called simple delta_delay timing. This simple model involves no explicit timing values.
Only the functional are modeled.It is purely functional model.
entity registre is
port(di: in bit_vector(1 to 8); strb: in BIT; ds1,nds2: in bit; do: out bit_vector(1 to 8));
end registre ;
architecture delta_delay_arch of registre is
signal reg: bit_vector(1 to 8);
delta_delay_proc:
process(strb,ds1,nds2)
if strb='1' and not strb'stable then
reg <= di;
if ds1='1' and nds2='0' then
do <= di;
else
do <= "11111111";
elsif
not ds1'stable or not nds2'stable then
do <= reg;
end process delta_delay_proc;
end delta_delay_arch;
Second implementation of registre implies a simplified timing. In this example, the temporal response on the input events are independent.
This allows to build an architecture with a single proces.
For example if strobe signal is validated than the input value is registered in the latch after strb_del (strobe delay).
If both the strobe and output are validated the input value appears at the output after the cumulated delay of strb_del plus out_del.
Note that the enable delay (enbl_del) necessary to activate the output validation is not taken into account.
architecture simple_delay of registre is
constant strb_del:TIME:=40ns;
constant out_del:TIME:=60ns;
constant enbl_del:TIME:=20ns;
simple_delay_proc:
reg <= di after strb_del ;
do <= di after strb_del+out_del;
do <= "11111111" after strb_del+out_del;
do <= reg after en_del+out_del;
do <= "11111111" after enbl_del+out_del;
end process simple_delay_proc;
end simple_delay;
The third model takes into account all events' combinations. This requires the introduction of three independent processes each one associated with one kind of correlated external and internal events.
The process strobe_proc operates on strobe signal and loads the latch.
The process enbld_proc operates on the output validation signals and generates an internal signal called enbld_sig.
The process output_proc is activated independently by new latch content and output validation signal.
architecture multi_path of registre is
signal enbld_sig: bit;
strobe_proc:
process(strb)
if strb='1' then
reg <= di after strb_del;
end process strobe_proc;
enbld_proc:
process(ds1,nds2)
enbld_sig <= ds1 and not nds2 after enbl_del;
end process enbld_proc;
output_proc:
process(reg,enbld_sig) -- both signals may activate the process in a short time and produce a spike
if enbld_sig ='1' then
do <= reg after out_del;
do <= "11111111" after out_del;
end process output_proc;
end multi_path;
The last implementation is conceptually the same as the third one. The only difference is in the way the three independent processes are created.
The activation of the first input process is conditionned by the use of a guarded signal.
The activation condition for this signal is defined in the block statement to which the guarded signal belongs.
The processes enbl_proc and output_proc are described as independed signal assignments.
architecture data_flow of registre is
strobe_block:
block(strb='1' and not strb'stable)
reg <= guarded di after strb_del; -- strobe activated process
enbld_sig<= ds1 and not nds2 after enbl_del; -- enable process
do <= reg after out_del when enbld_sig ='1' else "11111111" after out_del; -- output process
end block strobe_block;
end data_flow;
Test unit:
entity test_reg is -- verification of registre entity
end test_reg;
architecture simple of test_reg is
signal d: bit_vector(1 to 8);
signal s,pe,ne:bit;
component registre
end component ;
for iR: registre use entity work.REGISTRE(NODAL);
iR: registre port map(d,s,pe,ne,open);
process
d <="10101010","11110000" after 30ns,
"00001111" after 70ns, "10110100" after 125ns,
"01000111" after 180ns;
s<='1' after 20ns,'0' after 50ns, '1' after 80ns;
ne<='1' after 50ns, '0' after 80ns,'1' after 100ns ;
pe<='1' after 10ns, '0' after 120ns,'1' after 150ns;
end simple ;