#### VHDL Primer Tutorial #2 Mike Goldsmith Feb 3, 2004, ~ 2 hr duration ### Outline - IEEE 1164 and Built-In Data types - Arithmetic and Logic operators - More VHDL Syntax - Modularization and Instantiation - Test benches ## IEEE 1164 Data types • std\_ulogic ``` - 'U' => Uninitialized – '0' => Strong (forced) Zero - '1' => Strong One - 'X' => Strong Unknown => High Impedance - 'Z' - 'L' => Weak Zero - 'H' => Weak One => Weak Unknown - 'W' => Don't Care ``` SEE: http://www.ecs.soton.ac.uk/~ajr1/vhdl\_faq/std\_logic\_1164.html for gory details. # Built-in Data types - Bit '0', '1' - Boolean true, false - Integer integer numbers, eg: 25 - Real floating point numbers, eg: 2.57 - Time an integer value + unit, eg: 5 ms - Time has units of fs, ps, ns, us, ms, sec, min, hr - Character ASCII char set # Arithmetic Operators - + addition - - subtraction - / division - \* multiplication - \*\* exponential - mod modulus - rem remainder - abs absolute value # Logic Operators - not negation - and logical and - or logical inclusive or - xor logical exclusive or - nand negated logical and - nor negated logical inclusive or - xnor negated logical exclusive or # Comparison Operators - = equals - /= inequality - <= less than or equals - >= greater than or equals - < less than - > greater than - Conditional and Looping constructs **must** be within processes - Conditional Statements - If-then constructs - Switch / Case constructs - 'Condensed' processes (when construct) - Loops - Simple loops - While loops - For loops - If-Then: basic conditional, if 'a' then 'b' - Sample code: ``` [if_label.] if condition then --statements elsif alt_condition then --statements else --statements end if [if_label]; ``` - Switch / Case because writing 'elsif' 55 times really sucks. - Sample code: ``` [case_label:] case signal_name is when value_1 => --if sig = value_1 then --statements when value_n => --elsif sig = value_n then when default => --else end case [case label]; ``` - Condensed conditional processes: write a conditional process on one line - Sample code: ``` signal_1 <= signal_2 when condition else signal_3; Replaces: process( signal 2, signal 3, ...) is begin if condition then signal_1<= signal_2; else signal_1 <= signal 3; end if; end process; ``` - Simple loops: repeat a sequence of statements multiple times. - Sample code: ``` [/oop_label:] loop --statement(s) exit [/oop_ label] [when condition]; next [/oop_ label] [when condition]; --conditionally executed statement(s) end loop [/oop_ label]; ``` - While loops: execute loop while exit conditions are unmet. - Sample code: ``` [/oop_labe/:] while condition loop --statement(s) next [/oop_ labe/] [when condition]; --conditionally executed statement(s) end loop [/oop_ labe/]; ``` - For loops: execute loop a fixed number of times - Sample code: ``` [/oop_labe/:] for index in range loop --statement(s) next [/oop_ labe/] [when condition]; --conditionally executed statement(s) end loop [/oop_ labe/]; ``` Loop index is a variable with scope limited to the loop - Sequential (clocked) processes - Sample code: - Sequential processes (again) - Sample code: - Sequential processes (yet again) - Sample code: ``` [process_label:] process is begin wait until clk'event and clk = '1' q <= d; --simple D flip-flop, notice no --'else' case end process [process_ label]; ``` • Processes with 'wait' statements cannot have sensitivity lists - How to make one module talk to another - All modules are instantiated by other modules; the entire design falls under a 'top-level' module - The *interface* of a module must be defined for that module to be used. The *implementation* of the modules is selectable • Source code: ``` architecture arch_name of entity_name is component comp_name is inport. in type; port( outport. out type end component comp_name; begin --statements ``` • Source code: ``` begin [inst_label:] comp_name port map( inport => signal_1, outport => signal_2); --statements end architecture arch_name; ``` • Example: ``` architecture foo of bar is component inv is port( d : in std_logic; q: out std logic end component inv; signal s_in, s_out : std_logic; begin my_inverter: inv port map( d => s_in, q => s_out); --statements end architecture foo; ``` - Used for simulation and verification - Entity has no ports - Architecture instantiates **one** main module to be tested, plus optionally support modules - Module to be tested referred to as device under test (dut) or unit under test (uut) • Sample code: ``` entity comp_name_tb is end entity comp_name_tb; architecture test_name of comp_name_tb is component comp_name is ... begin uut: comp_name port map( ... ); ``` - Test benches use control and status signals to force operating conditions on the UUT and monitor the results - Test benches can be executed in simulation and results displayed on a waveform viewer - Test benches can also interact with the computer system, including file reading and writing, display to standard output, etc. • Example: entity int tb is end entity inv\_tb; architecture tb of inv tb is component inv is port( d : in std logic; q: out std logic end component inv; signal t\_in : std\_logic := '0'; signal t\_out : std\_logic; • Example: ``` begin uut: inv port map( d => t_in, q => t_out); t_in <= not t_in after 20 us; --create a 50 kHz clk end architecture tb;</pre> ``` • Test bench **must** have some form of signal that changes with time