## **Enhancements to VHDL's Packages**

Jim Lewis, Director of Training, SynthWorks Design Inc, Tigard, OR

# Abstract

VHDL is a strongly typed language. Success in VHDL depends on understanding the types and overloading of operators provided in the packages std\_logic\_1164 (IEEE standard 1164) and Numeric\_Std (IEEE standard 1076.3).

*Currently, enhancements for both of these packages are being finalized for the next drafts of these standards. These enhancements include:* 

- logical reduction functions,
- array/scalar logic operations,
- *array/scalar addition operators,*
- TO\_X01, TO\_X01Z, TO\_UX01, IS\_X for unsigned and signed,
- *shift operators for std\_logic\_vector and std\_ulogic\_vector*
- unsigned arithmetic for std\_logic\_vector and std\_ulogic\_vector (new package),
- *textio for types in std\_logic\_1164 and numeric\_std (two new packages),*
- floating point arithmetic (new packages)

This paper will provide details about the new features of std\_logic\_1164 and numeric\_std and also provides some rules of thumb for remembering the overloading. This paper also contains a minitutorial on these packages to help those who are not currently using numeric\_std to make the transition.

# New Features for Std\_Logic\_1164 and Numeric\_Std

This section contains proposals for additions to the packages std\_logic\_1164, numeric\_std, and numeric\_bit. Although numeric\_bit is not mentioned in the discussion, everything that applies to numeric\_std also applies to numeric\_bit.

#### Logical reduction operators: std\_logic\_1164, numeric\_std

Add reduction operators for std\_logic\_vector, std\_ulogic\_vector, unsigned and unsigned. Functions for **and\_reduce**, **nand\_reduce**, **or\_reduce**, **nor\_reduce**, **xor\_reduce**, and **xnor\_reduce** will be defined of the following form:

function and\_reduce ( arg : std\_logic\_vector ) return std\_ulogic;

This will allow expressions such as the following to be written:

Parity <= xor\_reduce (Data) and ParityEnable ;

#### Array/scalar logic operators: std\_logic\_1164, numeric\_std

Overload the logic operators to support mixing an array with a scalar for std\_logic\_vector, std\_ulogic\_vector, unsigned and signed. Functions for **and**, **nand**, **or**, **nor**, **xor**, and **xnor** will be defined in the following form:

function "and" (1: std\_logic\_vector; r : std\_ulogic ) return std\_logic\_vector; function "and" (1: std\_ulogic; r : std\_logic\_vector ) return std\_logic\_vector;

This solves the following common design problem:

signal ASel, BSel, CSel, DSel : std\_logic ;
signal Y, A, B, C, D : std\_logic\_vector(7 downto 0) ;
...
Y <= (A and ASel) or (B and BSel) or (C and CSel) or (D and DSel) ;</pre>

Without these functions, a common issue is to write the above code as follows:

Y <= A when ASel = '1' else B when BSel = '1' else C when CSel = '1' else D when DSel = '1' else (others => '0');

When the select signals (ASel, ...) are mutually exclusive, this hardware functions correctly. However, this code implies priority select logic and is inefficient from a hardware area and timing perspective.

#### Array/scalar addition operators: numeric\_std

Overload the addition operators to support mixing an array with a scalar for unsigned and signed. Functions for "+" and "-" will be defined in the following form:

function "+"(L: unsigned; R: std\_ulogic) return unsigned; function "+"(L: std\_ulogic; R: unsigned) return unsigned;

These functions facilitate writing the following add with carry in:

signal Cin : std\_logic ; signal A, B : unsigned(7 downto 0) ; signal Y : unsigned(8 downto 0) ;  $\dots$ Y <= A + B + Cin ;

They also facilitate writing the following conditional incrementer:

| n : std_log          | gic ;           |
|----------------------|-----------------|
| ur, IncNext : unsign | ed(7 downto 0); |
| ur, IncNext : unsign | ed(7 downto (   |

IncNext <= IncCur + IncEn ;</pre>

#### TO\_X01, TO\_X01Z, TO\_UX01, IS\_X: numeric\_std

Add the strength reduction and 'X' detection operators for unsigned and signed. Currently these operators are only supported for std\_logic\_vector and std\_ulogic\_vector. Functions for TO\_X01, TO\_X01Z, TO\_UX01, and IS\_X will be defined in the following form:

function To\_X01 (s : unsigned) return unsigned; function Is\_X (s : unsigned) return boolean; These functions are useful for testbenches to handle 'X's and resistive strength driving levels. It is also appropriate to use these functions in input pad cells of ASIC and FPGA libraries. In an RTL design, logic should only generate the values '0', '1', '-', and 'X'.

## Shift operators: std\_logic\_1164, numeric\_std?

Overload shift operators for std\_logic\_vector and std\_ulogic\_vector. Functions for sll, srl, sla, sra, rol, and ror will be defined in the following form:

function "sll" (1: std\_logic\_vector; r : integer ) return std\_logic\_vector; function "sll" (1: std\_ulogic\_vector; r : integer ) return std\_ulogic\_vector;

Numeric\_std currently supports sll, srl, rol, and ror. Support is being considered for sla and sra.

#### Unsigned arithmetic for std\_logic\_vector and std\_ulogic\_vector: new package

Create a new package that implements unsigned arithmetic operators for std\_logic\_vector and std\_ulogic\_vector. Tenatively the package is named numeric\_unsigned. It will include all functions included in numeric\_std minus the ones that are in std\_logic\_1164 (or planned for std\_logic\_1164).

A testbench is one of the places that will benefit most. Testbenches often need to perform a numeric algorithm on an object that is not numeric in a formal sense. For example, the following code shows data being written to consecutive bits in a RAM with exactly one bit set in each data word.

For RTL design the existence of this package permits one of three methodology variations:

| 1) Strict:      | Use only types unsigned and signed. Do not use numeric_unsigned. |
|-----------------|------------------------------------------------------------------|
| 2) Semi-Strict: | Use unsigned and signed for all math operations except counters. |
| 3) Flexible:    | Use std_logic_vector for any unsigned operation.                 |
|                 | Use signed for all signed operations.                            |

Note, this proposal does not include a numeric\_signed package. Use ieee.numeric\_std.signed for signed operations.

#### *Textio for std\_logic\_1164 and numeric\_std types: (two new packages)*

Overload read and write procedures to support std\_ulogic, std\_logic, std\_ulogic\_vector, std\_logic\_vector, unsigned, and signed. Functions for read and write will be defined in the following forms:

procedure READ( L: inout LINE; VALUE out std\_logic; GOOD: out BOOLEAN); procedure READ( L: inout LINE; VALUE out std\_logic); procedure WRITE( L: inout LINE; VALUE in std\_logic; JUSTIFIED: in SIDE:= RIGHT; FIELD: in WIDTH:= 0); Overload read and write procedures to support base operations with std\_ulogic\_vector, std\_logic\_vector, unsigned, and signed. Functions for read and write will be defined in the following forms:

type REPRESENTATION is ( any, binary, octal, hexadecimal ); procedure READ( L: inout LINE; VALUE out std\_logic; GOOD: out BOOLEAN; R: in REPRESENTATION := any ); procedure WRITE( L: inout LINE; VALUE in std\_logic; JUSTIFIED: in SIDE:= RIGHT; FIELD: in WIDTH:= 0; R: in REPRESENTATION := any);

#### Floating point arithmetic

A family of VHDL packages are being introduced to implement floating point arithmetic. For more information, paper [1] in this conference. Also see <u>http://www.eda.org/fphdl</u>.

# Strong Typing, Overloading, & Conversions

## Strong Typing

VHDL is a strongly typeed language. Basically this means that every value returned by an expression must be an appropriate size and type for the context in which it is being used. If A, B, and Y are signals of type ieee.numeric\_std.unsigned, then the following table applies:

| Expression                 | Size of Y                    |
|----------------------------|------------------------------|
| Y <= A ;                   | A'Length                     |
| $Y \leq A \text{ and } B;$ | A'Length = B'Length          |
| $Y \le A > B;$             | Boolean                      |
| $Y \le A + B;$             | Maximum (A'Length, B'Length) |
| $Y \le A + 10;$            | A'Length                     |
| Y <= A * B ;               | A'Length + B'Length          |

Strong typing provides a limited amount of error checking of expression construction. To see how this works, consider the following code.

| signal A8, B8, Result8                   | : unsigned (7 downto 0); |
|------------------------------------------|--------------------------|
| signal Result9                           | : unsigned (8 downto 0); |
| signal Result7                           | : unsigned (6 downto 0); |
|                                          |                          |
| $\operatorname{Result8} \leq = A8 + B8;$ |                          |
| Result9 <= $('0' \& A8) + ('$            | 0' & B8) ;               |
| Result7 $\leq$ A(6 downto 0)             | ) + B(6  downto  0);     |

For each expression above to be correct, the expression must be correctly sized to match the size of the result. If this requirement is not met, the code will not compile. This leads to an observation: With strong typing, on a bad day, you will be abused by the compiler. However, without strong typing, on a bad day, you can code errors into your design that will require a good testbench and lots of time to find.

## **Overloading**

Another great feature that strong typing facilitates is overloading of subprograms and operators. Overloading allows a subprogram name or operator symbol to be used more than once as long as there is a way to differentiate the calls. This means that when a new type, such as float, is added to a package, it can use the same set of operator symbols (+, -, \*, ...) that are used for all the other types.

In addition to understanding the previous set of expression rules, to be successful in VHDL, you must know about the overloading provided by the packages. The following tables summarize the overloading proposed for the enhanced versions of std\_logic\_1164, numeric\_std, and numeric\_unsigned.

| Operator        | Right             | Result     | Package        |
|-----------------|-------------------|------------|----------------|
| Logic reduction | Std_logic_vector  | Std_ulogic | Std_logic_1164 |
|                 | Std_ulogic_vector | Std_ulogic |                |
|                 | Unsigned          | Std_ulogic |                |
|                 | Signed            | Std_ulogic |                |

| Operator | Left              | Right                        | Result            | Package        |
|----------|-------------------|------------------------------|-------------------|----------------|
| Logic    | Std_ulogic        | Std_ulogic                   | Std_ulogic        | Std_logic_1164 |
|          | Std_logic_vector  | Std_logic_vector             | Std_logic_vector  |                |
|          | Std_logic_vector  | Std_ulogic                   | Std_logic_vector  |                |
|          | Std_ulogic        | Std_logic_vector             | Std_logic_vector  |                |
|          | Std_ulogic_vector | Std_ulogic_vector            | Std_ulogic_vector |                |
|          | Std_ulogic_vector | Std_ulogic_vector Std_ulogic |                   |                |
|          | Std_ulogic        | Std_ulogic_vector            | Std_ulogic_vector |                |
|          | Unsigned          | Unsigned                     | Unsigned          | Numeric_std    |
|          | Unsigned          | Std_ulogic                   | Unsigned          |                |
|          | Std_ulogic        | Unsigned                     | Unsigned          |                |
|          | Signed            | Signed                       | Signed            |                |
|          | Signed            | Std_ulogic                   | Signed            |                |
|          | Std_ulogic        | Signed                       | Signed            |                |

| Operator                                         | Left              | Right             | Result  | Package             |
|--------------------------------------------------|-------------------|-------------------|---------|---------------------|
| Comparison                                       | Std_ulogic        | Std_ulogic        | boolean | *                   |
|                                                  | Std_logic_vector  | Std_logic_vector  | boolean | Numeric_unsigned, * |
|                                                  | Std_logic_vector  | Integer           | boolean | Numeric_unsigned    |
|                                                  | Integer           | Std_logic_vector  | boolean | Numeric_unsigned    |
|                                                  | Std_ulogic_vector | Std_ulogic_vector | boolean | Numeric_unsigned, * |
|                                                  | Std_ulogic_vector | Integer           | boolean | Numeric_unsigned    |
|                                                  | Integer           | Std_ulogic_vector | boolean | Numeric_unsigned    |
|                                                  | Unsigned          | Unsigned          | Boolean | Numeric_std         |
|                                                  | Unsigned          | Natural           | boolean | Numeric_std         |
|                                                  | Natural           | Unsigned          | boolean | Numeric_std         |
|                                                  | Signed            | Signed            | Boolean | Numeric_std         |
|                                                  | Signed            | Integer           | boolean | Numeric_std         |
|                                                  | Integer           | Signed            | boolean | Numeric_std         |
| Notes: * Implicitly created comparison operators |                   |                   |         |                     |

| Operator       | Left              | Right             | Result            | Package          |
|----------------|-------------------|-------------------|-------------------|------------------|
| Addition       | Std_logic_vector  | Std_logic_vector  | Std_logic_vector  | Numeric_unsigned |
|                | Std_logic_vector  | Integer           | Std_logic_vector  | Numeric_unsigned |
|                | Integer           | Std_logic_vector  | Std_logic_vector  | Numeric_unsigned |
|                | Std_logic_vector  | Std_ulogic        | Std_logic_vector  | Numeric_unsigned |
|                | Std_ulogic        | Std_logic_vector  | Std_logic_vector  | Numeric_unsigned |
|                | Std_ulogic_vector | Std_ulogic_vector | Std_ulogic_vector | Numeric_unsigned |
|                | Std_ulogic_vector | Integer           | Std_ulogic_vector | Numeric_unsigned |
|                | Integer           | Std_ulogic_vector | Std_ulogic_vector | Numeric_unsigned |
|                | Std_ulogic_vector | Std_ulogic        | Std_ulogic_vector | Numeric_unsigned |
|                | Std_ulogic        | Std_ulogic_vector | Std_ulogic_vector | Numeric_unsigned |
|                | Unsigned          | Unsigned          | Unsigned          | Numeric_std      |
|                | Unsigned          | Natural           | Unsigned          | Numeric_std      |
|                | Natural           | Unsigned          | Unsigned          | Numeric_std      |
|                | Unsigned          | Std_ulogic        | Unsigned          | Numeric_std      |
|                | Std_ulogic        | Unsigned          | Unsigned          | Numeric_std      |
|                | Signed            | Signed            | Signed            | Numeric_std      |
|                | Signed            | Integer           | Signed            | Numeric_std      |
|                | Integer           | Signed            | Signed            | Numeric_std      |
|                | Signed            | Std_ulogic        | Signed            | Numeric_std      |
|                | Std_ulogic        | Signed            | Signed            | Numeric_std      |
|                |                   |                   |                   |                  |
| Operator       | Left              | Right             | Result            | Package          |
| Multiplication | Std logic vector  | Std logic vector  | Std logic vector  | Numeric unsigned |

| Operator       | Left              | Right                           | Result            | Package          |
|----------------|-------------------|---------------------------------|-------------------|------------------|
| Multiplication | Std_logic_vector  | l logic vector Std logic vector |                   | Numeric_unsigned |
|                | Std_logic_vector  | Integer                         | Std_logic_vector  | Numeric_unsigned |
|                | Integer           | Std_logic_vector                | Std_logic_vector  | Numeric_unsigned |
|                | Std_ulogic_vector | Std_ulogic_vector               | Std_ulogic_vector | Numeric_unsigned |
|                | Std_ulogic_vector | Integer                         | Std_ulogic_vector | Numeric_unsigned |
|                | Integer           | Std_ulogic_vector               | Std_ulogic_vector | Numeric_unsigned |
|                | Unsigned          | Unsigned                        | Unsigned          | Numeric_std      |
|                | Unsigned          | Natural                         | Unsigned          | Numeric_std      |
|                | Natural           | Unsigned                        | Unsigned          | Numeric_std      |
|                | Signed            | Signed                          | Signed            | Numeric_std      |
|                | Signed            | Integer                         | Signed            | Numeric_std      |
|                | Integer           | Signed                          | Signed            | Numeric_std      |

| Operators       | Left       | Right      | Result     | Notes:                                |
|-----------------|------------|------------|------------|---------------------------------------|
| Logic           | ТуреА      | ТуреА      | ТуреА      | Array = unsigned, signed,             |
|                 |            |            |            | std ulogic vector, std logic vector   |
| Numeric         | Array      | Array      | Array, *   | $TypeA = boolean std_logic$           |
|                 | Array      | Integer    | Array, *   | std ulogic. Array                     |
|                 | Integer    | Array      | Array, *   | * for comparison operators the result |
|                 |            |            |            | is boolean                            |
| Logic, Addition | Array      | Std_ulogic | Array      | 15 boolean                            |
|                 | Std_ulogic | Array      | Array      |                                       |
|                 |            |            |            |                                       |
| Logic Reduction |            | Array      | Std_ulogic |                                       |

The following is a generalization of the previous tables.

#### **Type Conversions**

In addition to understanding the available types and overloading, it is also important to know how to convert between std\_ulogic, std\_logic, unsigned, signed, std\_logic\_vector and integer. Some of these can be accomplished by built in language features and some by conversion fuctions in numeric\_std.

Conversion between std\_logic and std\_ulogic occurs automatically. In VHDL, two types convert automatically when they are both subtypes of the same type. Std\_logic is a subtype of std\_ulogic and std\_ulogic is a subtype of itself. Elements of signed, unsigned, and std\_logic\_vector are all of the type std\_logic and convert automatically to std\_ulogic. As a result, in the example below, only the "AND" function with std\_ulogic arguments is required.

signal A\_sl : std\_logic ; signal B\_slv : std\_logic\_vector(7 downto 0) ; signal C\_uv : unsigned (7 downto 0) ; signal D\_sv : signed (7 downto 0) ; signal Y\_sul : std\_ulogic ;

 $Y_sul \le A_sl and B_slv(0) and C_uv(1) and D_sv(2)$ ;

Conversion between std\_logic\_vector, signed, and unsigned can be accomplished by type casting. In VHDL, type casting can be used to convert two equivalent sized arrays when they both have a common base type (std\_logic) and their indicies have a common base type (natural). Hence, to subtract two unsigned numbers and get a signed result:

signal A\_uv, B\_uv : unsigned (7 downto 0); signal Y\_sv : signed (8 downto 0); .... Y\_sv <= signed('0' & A\_uv) - signed('0' & B\_uv);</pre> To convert from unsigned and signed to integer use the to\_integer function from numeric\_std as shown in the example below:

| signal A_uv, C_uv   | <pre>: unsigned (7 downto 0);</pre> |
|---------------------|-------------------------------------|
| signal Unsigned_int | : integer range 0 to 255;           |
| signal B_sv, D_sv   | : signed( 7 downto 0);              |
| signal Signed_int   | : integer range -128 to 127;        |
| Unsigned_int        | <= TO_INTEGER ( A_uv );             |
| Signed_int          | <= TO_INTEGER ( B_sv );             |

To convert from integer to unsigned and signed use the to\_unsigned and to\_signed functions from numeric\_std as shown in the example below. Note the value 8 specifies the width of the array.

C\_uv <= TO\_UNSIGNED ( Unsigned\_int, 8) ; D\_sv <= TO\_SIGNED ( Signed\_int, 8 ) ;

To convert between std\_logic\_vector and integer, use type casting plus type conversion fuctions.

signal E\_slv : std\_logic\_vector (7 downto 0); signal Unsigned\_int : integer range 0 to 255; .... E slv <= std logic vector( to unsigned(Unsigned int, E slv'length));</pre>

# **Participating In Standards**

VHDL standards are IEEE standards. As a VHDL community member it is both your right and responsibility to join IEEE committees and participate in VHDL standards. If you don't participate, the changes you envision and wish for (no matter how simple or obvious) will not happen. To find out more about participating in VHDL standards go the the web links, <u>http://www.eda.org</u> and <u>http://www.SynthWorks.com/VhdlLinks.htm</u>.

## Acknowlegements

This paper documents work which is the collaboration of the participants of the IEEE 1076.3 and IEEE 1164 reflectors.

## References

- 1. Bishop, David "Floating Point for VHDL and Verilog", to be published in DVCon 2003 proceedings.
- 2. Lewis, Jim "Comprehensive VHDL Introduction" copyright by SynthWorks Design Inc, 1999 through 2003. Numerous examples used by permission.

# About SynthWorks VHDL Training

Comprehensive VHDL Introduction 4 Days

http://www.synthworks.com/comprehensive\_vhdl\_introduction.htm Engineers learn VHDL Syntax plus basic RTL coding styles and simple procedure-based, transaction testbenches. Our designer focus ensures that your engineers will be productive in a VHDL design environment.

VHDL Coding Styles for Synthesis 4 Days <u>http://www.synthworks.com/vhdl\_rtl\_synthesis.htm</u> Engineers learn RTL (hardware) coding styles that produce better, faster, and smaller logic.

VHDL Testbenches and Verification 3 days <u>http://www.synthworks.com/vhdl\_testbench\_verification.htm</u> Engineers learn how create a transaction-based verification environment based on bus functional models.

For additional courses see: <u>http://www.synthworks.com</u>