# EE-334 Digital System Design

**Custom Digital Circuits** 

VHDL for Synthesis – Basic Constructs and Correspondences

**Andreas Burg** 

#### Hierarchy and Instantiation

- Hierarchy is required for modularity, abstraction, and to deal with complexity
- Blocks used later in a hierarchy are called COMPONENTS
  - Components can themselves be hierarchical (include instances of other components)
  - Signals declared in a component are only visible within the component
  - The interface of a component is defined by its ENTITY
- Each "appearance/use" of a component is called an INSTANCE
  - There can be multiple instances of the same component
  - Instances are connected through the ports of their components



#### Components vs. Functions

Only distant analogy to functions in programming: similar purpose (reuse),
 BUT very different physical presence

#### **Hardware (VHDL)**





Co-exist in parallel

#### Software (C)

```
Int my_fct(int a, int b)
{
    return a+b;
}

Main() {

    x = my_fct(a, b);
    y = my_fct(c, d);
    z = my_fct(x, y);
}
Function
calls
```

Sequential execution

#### Instantiating Components in VHDL (1/2)

- Components need to be declared before instantiating/using them in the architecture of another component
  - Declaration in the preamble of the architecture in which they are used
  - Alternative: declaration in a package (not discussed here)
- Component declaration & name must match the corresponding entity name

Entity declaration of a component in a file, e.g., entity\_name.vhd

```
ARCHITECTURE architecture name OF other entity name IS
         -- component declaration
         COMPONENT component name IS
              GENERIC (
                   generic_1_name : generic_1_type;
                   generic 2 name : generic 2 type
              PORT
                   port 1 name : port 1 dir port 1 type;
                   port 2 name : port 2 dir port 2 type
         END COMPONENT;
BEGIN
         -- VHDL statements
END architecture name;
```

#### Instantiating Components in VHDL (2/2)

Declared components can be instantiated in the architecture body

#### Instantiation(s)

- defines the instance name
- connects ports of an instance of the components to signals
- Defines the generics (parameters) based on expressions that can be evaluated at compile-time

```
ARCHITECTURE architecture name OF other entity name IS
         -- signal declarations
         SIGNAL port_1_1_signal, port_2_1_signal : port_1_type;
         SIGNAL port 1 2 signal, port 2 2 signal : port 2 type;
         -- component declaration
         COMPONENT component_name IS
              GENERIC ( ... );
              PORT ( ... );
         END COMPONENT;
BEGIN
         -- Component instantiation
         instance 1 name : component name
              GENERIC MAP (
                   generic 1 name => CONSTANT EXP 1 1,
                   generic_2_name => CONSTANT_EXP_1_2
              PORT MAP (
                   port 1 name => port 1 1 signal,
                   port 2 name => port 1 2 signal
         instance 2 name : component name
              GENERIC MAP ( CONSTANT EXP 2 1, CONSTANT EXP 2 2
              PORT MAP ( port 2 1 signal, port 2 2 signal );
END architecture name;
```



#### **Array Types**

- To describe hardware we often need BUSSES (groups of signals or constants)
- Arrays are defined by declaring a custom type
- Custom types are declared in the architecture preamble
  - Elements from left-to-right can be counted "from-low-to-high" or "from-high-downto-low"
  - Array types, once declared, can again serve as basis for new arrays to build 2+D arrays

Array size declaration can be deferred until type is used



#### STD\_LOGIC\_VECTOR with DOWNTO Index

- STD\_LOGIC\_VECTOR: heavily used in hardware modelling
  - Array type that comes with STD\_LOGIC type
  - Defined in IEEE.STD\_LOGIC\_1164 package
  - STD\_LOGIC\_VECTORS for binary numbers are typically declared "from-high-downto-low"
    - Correlates better to the established weighted number representation
      - MSB on the left (low index)
      - LSB on the right (high index)

```
"0 1 0 0 0" = 8'dec
Bit idx. 4 3 2 1 0
(4 downto 0)
```

Ideally, use only DOWNTO

#### Operations on Array Types

Accessing from and assigning to array elements or ranges

```
target_array_object(index) <= base_type_object
target_base_type_object <= target_base_type_array(index)
target_array_object(index_range) <= from_array_object(index_range)
index_range = low TO high | high DOWNTO low</pre>
```

- Assigning array aggregates (collections of elements) to an array
  - Multiple options exist to "fill" an array (assign multiple elements together)

```
target_array_object <= (value_1, value_2, ...);
target_array_object <= (idx_1=>value_1, idx_2=>value_2, ...);
target_array_object <= (idx_1=>value_1, idx_2|idx_3=>value_2, ...);
```

#### Operations on Array Types

• Filling an array: OTHERS refers to all still unassigned elements in the aggregator

```
target_array_object <= (idx_1=>value_1, OTHERS=>value_2); -- fill all remaining
target_array_object <= (OTHERS=>value_1); -- fill all elements
```

- Assignments to arrays with character elements
  - Array literal values can be places in double quotes

```
target_array_object(index_range) <= "..." -- e.g., "0100-1"</pre>
```

• STD\_LOGIC\_VECTOR is based on STD\_LOGIC which is a character type, i.e., '0', '1', 'X', '-', ...

```
target_array_object(index_range) <= "010-10-"</pre>
```

Concatenation of arrays and array elements

```
target_array_object(index_range) <= base_type_object_1 & base_type_object_1</pre>
```



#### Example Operations on STD\_LOGIC\_VECTOR

Signal declarations of STD\_LOGIC\_VECTOR

```
-- signal declaration of std_logic_vectors
SIGNAL AxD, BxD, CxD, DxD, ExD : STD_LOGIC_VECTOR(8-1 DOWNTO 0);
SIGNAL QxS : STD_LOGIC;
```

Assignments

```
QxS <= '1';
AxD <= "10010101";
BxD <= "-0-001-1";
CxD <= (OTHERS => '0');
```

Concatenation and indexing

#### Types for Arithmetic: UNSIGNED/SIGNED

- Built-in type INTEGER supports basic arithmetic operations,
   BUT the integer type is not sufficiently generic for optimized hardware
  - INTEGER type has a fixed width of 32 bit (often too large or sometimes too small)
  - INTEGER represents only signed numbers having half the range
  - Any overflow or underflow will trigger an ERROR in the simulation rather than a wrap-around
- Need for a more flexible type with variable length for signed and unsigned
- IEEE numeric\_std package: define integer as array of std\_logic
- Two new data types: UNSIGNED, SIGNED
- The array interpreted as an unsigned or signed binary numbers
  - UNSIGNED are represented as standard binary
  - SIGNED vectors are represented using two's complement
  - Array elements can be accessed and assigned as in a std\_logic\_vector



#### UNSIGNED/SIGNED Data Types (Best practice)

- SIGNED and UNSIGNED data types represent numbers. Therefore
  - Corresponding signals best carry the suffix xD to indicate the numeric type of data
     <Signal Name>xD

MyInputAxDI, MySIGNALxD, AccuREGxDN, AccuREGxDP

- SIGNED and UNSIGNED use weighted binary digits, counted from right to left
  - Use DOWNTO bit order to place the LSB (bit-0) on the right and the MSB (bin-N) on the left

```
SIGNAL <Signal Name>xD : UNSIGNED(8-1 DOWNTO 0);
SIGNAL ExSIGNALxD : SIGNED(8-1 DOWNTO 0);
```

- When declaring a signal, it is often useful to immediately see the number of bits
  - Since bits are counted from zero (0), it is often more clear to write

```
SIGNAL <Signal Name>xD : UNSIGNED(#BITS-1 DOWNTO 0);
```



#### Arithmetic Operations on SIGNED and UNSIGNED

#### • IEEE numeric\_std defines many common operators

| overloaded<br>operator                                 | description                | data type<br>of operand a                                  | data type<br>of operand b                                  | data type<br>of result                   |
|--------------------------------------------------------|----------------------------|------------------------------------------------------------|------------------------------------------------------------|------------------------------------------|
| abs a<br>– a                                           | absolute value<br>negation | signed                                                     |                                                            | signed                                   |
| a * b<br>a / b<br>a mod b<br>a rem b<br>a + b<br>a - b | arithmetic<br>operation    | unsigned<br>unsigned, natural<br>signed<br>signed, integer | unsigned, natural<br>unsigned<br>signed, integer<br>signed | unsigned<br>unsigned<br>signed<br>signed |
| a = b<br>a /= b<br>a < b<br>a <= b<br>a > b<br>a >= b  | relational<br>operation    | unsigned<br>unsigned, natural<br>signed<br>signed, integer | unsigned, natural<br>unsigned<br>signed, integer<br>signed | boolean<br>boolean<br>boolean<br>boolean |



#### Arithmetic Operations on SIGNED and UNSIGNED

IEEE numeric\_std defines many common operators and type conversions

| function                                                                       | description                                              | data type of operand a                                                 | data type of operand b | data type of result           |
|--------------------------------------------------------------------------------|----------------------------------------------------------|------------------------------------------------------------------------|------------------------|-------------------------------|
| <pre>shift_left(a,b) shift_right(a,b) rotate_left(a,b) rotate_right(a,b)</pre> | shift left<br>shift right<br>rotate left<br>rotate right | unsigned, signed                                                       | natural                | same as a                     |
| resize(a,b)<br>std_match(a,b)                                                  | resize array<br>compare '-'                              | unsigned, signed<br>unsigned, signed<br>std_logic_vector,<br>std_logic | natural<br>same as a   | same as a<br>boolean          |
| to_integer(a) to_unsigned(a,b) to_signed(a,b)                                  | data type<br>conversion                                  | unsigned, signed<br>natural<br>integer                                 | natural<br>natural     | integer<br>unsigned<br>signed |

#### Type Conversions to/from SIGNED and UNSIGNED

- STD\_LOGIC\_VECTOR, UNSIGNED, and SIGNED are defined as arrays of std\_logic
- However, they are considered as different types
- Type conversion functions needed between these types
  - Second "length" argument required for source types without explicit length (INTEGER)
- Type conversion requires
   NO hardware resources

| From             | То               | Cast/Function                    |
|------------------|------------------|----------------------------------|
| std_logic_vector | unsigned         | unsigned(std_logic_vector)       |
| std_logic_vector | signed           | signed(std_logic_vector)         |
| unsigned         | std_logic_vector | std_logic_vector(unsigned)       |
| unsigned         | signed           | singed(unsigned)                 |
| signed           | std_logic_vector | std_logic_vector(signed)         |
| signed           | unsigned         | unsigned(signed)                 |
| unsigned         | integer          | to_integer(unsigned)             |
| signed           | integer          | to_integer(signed)               |
| integer          | unsigned         | to_unsigned(integer, no_of_bits) |
| integer          | signed           | to_signed(integer, no_of_bits)   |

#### Arithmetic Operations Example

Basic Arithmetic Operations: ENTITY ports are std\_logic\_vector

```
LIBRARY ieee;
USE ieee.std logic 1164.ALL;
USE ieee.numeric std.all;
ENTITY adder IS
     PORT (AxDI, BxDI
                             : IN std logic vector(8-1 downto 0);
                              : OUT std logic vector(8-1 downto 0));
            CxD0
END adder;
ARCHITECTURE rtl OF adder IS
          -- signal declaration
          SIGNAL SgnAxD, SgnBxD, SgnCxD : signed(8-1 DOWNTO 0);
                                                                                        No
BEGIN
                                                                                    hardware
          -- Type conversion
           SgnAxD <= signed(AxDI); ]</pre>
                                                                                    resources
           SgnBxD <= signed(BxDI); [</pre>
          -- Arithmetic
           SgnCxD <= SgnAxD + SgnBxD;</pre>
          -- Convert back to std logic vector for output port
           CxD0 <= std_logic_vector(SgnCxD); 4</pre>
END rtl;
```

#### The Problem of Describing Logic Efficiently

 Combinational logic is the work-horse of any digital circuit as it performs the data manipulation (algorithm/operations)

 Truth/look-up tables are the most generic way to describe Boolean logic, but also the most tedious one: difficult to formulate and to read/interpret



- Some more efficient means exist to describe specific types of logic:
  - Sometimes operations are naturally described in Boolean equations, but these quickly become difficult to create or understand
  - Arithmetic operations describe a frequently used specific subset of operations and we know how to map these to Boolean logic (from many early research works)

Need an efficient way to describe Boolean logic

#### Efficiently Describing Logic as Decision Trees

- A fully arbitrary logic function must ultimately be described by a truth/look-up table
- Completely arbitrary functions are rare in practice
  - Some inputs or combinations of some inputs render the output independent of all other inputs
  - In Karnaugh maps, we refer to this as "combining minterms"
- Such truth tables can be expressed efficiently and interpreted as decision diagrams
  - Interpretation as decision trees
  - Formulation based on if-then-elsif-elsif-...-else

|            | 00 | 01 | 11 | 10 |
|------------|----|----|----|----|
| 00         | 1  | 0  | 0  | 1  |
| 01         | 0  | 0  | 1  | 0  |
| <b>1</b> 1 | 0  | 0  | 0  | 0  |
| <b>1</b> 0 | 0  | 0  | 0  | 0  |
|            |    |    | _  |    |

#### Designing Logic "with/as Multiplexers" (1/2)

- Besides boolean and arithmetic operations, the CONDITIONAL ASSIGNMENT is a fundamental building block for HDL-based hardware design
- The simplest conditional assignment corresponds to a 2-input MUX
  - natural way to combine 2 pieces of logic into one

## HDL Pseudo-Code Conditional Assignment



Hardware



## Designing Logic "with/as Multiplexers" (2/2)

How to describe the combination of multiple expressions?

Nested IF statements: VHDL version described later

```
If boolean_expression_1 then
signal_1 <= expression_2

If boolean_expression_2 then
signal_1 <= expression_2

Else

signal_1 <= expression_3

End

End

Nesting level
```

- Nesting level
  - encodes priority in the HDL description
  - defines the level of the MUX in the HW
    - Outer-most IF corresponds to last MUX in the chain
    - Inner-most IF corresponds to first MUX in the chain



## Designing Logic "with/as Multiplexers" (2/2)

How to describe the combination of multiple expressions?

Combined IF, ELSIF, ELSIF, ..., ELSE statements:

```
If boolean_expression_1 then
signal_1 <= expression_1
Elsif boolean_expression_2 then
signal_1 <= expression_2
Else
signal_1 <= expression_3
End

Order
```

- Order
  - encodes priority in the HDL description
  - defines the level of the MUX in the HW
    - First IF corresponds to last MUX in the chain
    - Last IF corresponds to first MUX in the chain



### Conditional Assignments (MUXes) in VHDL (1/2)

Due to their importance and power to describe logic in an intuitive manner,
 VHDL provides many options for conditional assignments

**ELSE** 

WHEN-ELSE statement for multiple different binary conditions (IF, ELSIF, ELSIF, ... ELSE)

Multiple conditions may be true at the same time

- Order encodes the priority: only first one is relevant
- Note: often expression\_x is simply a signal





boolean

expression

signal 1

expression

Expression

## Conditional Assignments (MUXes) in VHDL (2/2)

Due to their importance and power to describe logic in an intuitive manner,
 VHDL provides many options for conditional assignments

Hardware
WITH-SELECT statement for a single multi-valued (non-binary) condition

```
WITH cond_signal SELECT
  target_signal <=
  expression_1 WHEN constant_1,
  expression_2 WHEN constant_2,
...
  expression_N WHEN OTHERS;
  ELSE</pre>
```

- Only one condition is true at the same time
- Note: often expression\_x is simply a signal



#### An Important Remark on Conditional Assignments

Consider the pseudo-code below, which is common practice in Software

NOTE: If none of the conditions is met,
 no assignment is made to signal\_1

 HOWEVER, a physical wire can never be "not assigned" any value at all



#### An Important Remark on Conditional Assignments

Consider the pseudo-code below, which is common practice in Software

NOTE: If none of the conditions is met,
 no assignment is made to signal\_1

- HOWEVER, a physical wire can never be "not assigned" any value at all
- In VHDL, signals preserve their state if no value is assigned
  - Works perfectly in simulation, BUT
    - Is often not the desired behaviour
    - Is NOT COMPATIBLE with the rules of synchronous design => issues later in the design



RULE: every combinational conditional assignment must be complete



#### Example: A Simple ALU

- Specification: 8-bit ALU
  - Three operations: +, -, AND
  - Specified by CMDxSI (2-bit command)
  - For CMDxDI="11", the output DOES NOT MATTER

```
LIBRARY ieee;
USE ieee.std_logic_1164.ALL;
USE ieee.numeric_std.all
ENTITY my_first_counter IS

PORT (
    AxDI : IN std_logic_vector(8-1 DOWNTO 0);
    BxDI : IN std_logic_vector(8-1 DOWNTO 0);
    CMDxSI : IN std_logic_vector(2-1 DOWNTO 0);

CxDO : OUT std_logic_vector(8-1 DOWNTO 0)
);
END my_first_counter;
...
```



```
ARCHITECTURE rtl OF my_first_counter IS
-- signal declaration
SIGNAL SgnCxD : SIGNED(8-1 DOWNTO 0);

BEGIN

WITH CMDxSI SELECT
SgnCxD <=
SIGNED(AxDI) + SIGNED(BxDI) WHEN "00",
SIGNED(AxDI) - SIGNED(BxDI) WHEN "01",
SIGNED(AxDI AND BxDI) WHEN "10",
"-----" WHEN OTHERS;

-- Output assignment with type conversion
CxDO <= std_logic_vector(SgnCxD);
END rtl;
```

#### How to Implement Registers in a Clean Way?

- Synchronous designs requires the notion of a state
  - In VHDL, signals preserve their state (have a state) when nothing is assigned to them
  - This preservation of states can be exploited to describe registers, but it must be done with care
- Objective: stick to the rules of synchronous design
  - ONLY the clock triggers a state transition
  - Use ONLY positive edge triggered FlipFLops (no latches)
  - → Incomplete combinational conditional assignments are not the solution to create registers
- Clean solution:

Describe registers EXPLICITLY with a well controlled template



#### Describing Edge Triggered Registers in VHDL

- Two ingredients:
  - A conditional statement that is TRUE when an edge (transition) occurs: clock\_signal'event
    - Positive edge: clock\_signal'event AND clock\_signal = '1'
  - A special process template that is well understood and clean
    - Assign input signal of a
       FlipFlop (next\_state\_signal)
       to the FlipFlop output signal
       (present\_state\_signal)
    - Conditional assignment is incomplete and only triggers on rising edge of the clock

```
ARCHITECTURE architecture name OF other entity name IS
         -- signal declaration
         SIGNAL next state signal 1, next state signal 2: state signal type;
         SIGNAL present_state_signal_1 : state_signal_type;
         SIGNAL present state signal 2 : state signal type;
BEGIN
         -- Clocked Process, generating a FlipFlop behavior
         p seq: PROCESS (clock signal) IS
         BEGIN
                -- process name: p seq
                   IF clock signal'EVENT AND clock signal = '1' THEN
                            present state signal 1 <= next state signal 1</pre>
     TWO
                                             expression;
                            present state signal 2 <= next state signal 2</pre>
   registers
                                             expression;
                   END IF;
         END PROCESS p seq;
END architecture name;
```

#### Describing Registers with Asynchronous Reset

- An asynchronous reset triggers also a state transition, independent of clock
  - The asynchronous reset typically takes precedence over the clock
    - Asynchronous reset (async\_reset\_signal)
      - assigns a constant to the
        FlipFlop output signal
        (present\_state\_signal)
    - Conditional assignment is still incomplete and now triggers on rising edge of the clock and on the reset signal
    - Reset can be low- or high-active

```
async_reset_signal
next_state_signal
clock_signal
```

```
ARCHITECTURE architecture name OF other entity name IS
         -- signal declaration
         SIGNAL next state signal : state signal type;
         SIGNAL present state signal : state signal type;
BEGIN
         -- Clocked Process, generating a FlipFlop behavior
         p seq: PROCESS (clock signal, async reset signal) IS
         BEGIN
                -- process name: p seq
                   IF async_reset_signal = '0|1' THEN
                             present state signal <= constant;</pre>
                   ELSIF clock_signal'EVENT AND clock_signal = '1' THEN
                             present state signal <= next state signal</pre>
                                                        expression;
                   END IF;
         END PROCESS p seq;
END architecture name;
```

#### Example: A Simple Counter (Overflowing)

Specification: 8-bit counter, overflowing (wrap-around)

```
LIBRARY ieee;
USE ieee.std_logic_1164.ALL;
USE ieee.numeric_std.all
ENTITY my_first_counter IS
PORT (
    CLKxCI : IN std_logic;
    RSTxRBI : IN std_logic;
    CNTxDO : OUT std_logic_vector(8-1 DOWNTO 0)
    );
END my_first_counter;
...
```

```
ARCHITECTURE rtl OF my first counter IS
          -- signal declaration
          SIGNAL CNTxDN : UNSIGNED(8-1 DOWNTO 0);
          SIGNAL CNTxDP : UNSIGNED(8-1 DOWNTO 0);
BEGIN
          -- Counting/incrementing (Combinational Logic)
          CNTxDN <= CNTxDP + 1;</pre>
          -- Clocked Process, generating a FlipFlop behavior
          p seq: PROCESS (CLKxCI, RSTxRBI) IS
          BEGIN -- process name: p seq
                   IF RSTxRBI = '0' THEN
                             CNTxDP \leftarrow (OTHERS => '0');
                    ELSIF CLKxCI'EVENT AND CLKxCI = '1' THEN
                             CNTxDP <= CNTxDN;</pre>
                    END IF;
          END PROCESS p seq;
          -- Output assignment with type conversion
          CNTxD0 <= std logic vector(CNTxDP);</pre>
END rtl;
```