#### TALLINN UNIVERSITY OF TECHNOLOGY

School of Information Technologies Thomas Johann Seebeck Department of Electronics

> [Thesis code] Gaspar Karm

# VHDL as Object-oriented Intermediate Language and Python bindings

Master's Thesis

#### Supervisors:

Muhammad Mahtab Alam PhD COEL ERA-Chair, Associate Professor

Yannick Le Moullec PhD Professor

# **Contents:**

| 1                         | VHI    | DL as i | ntermediate language                          |    |
|---------------------------|--------|---------|-----------------------------------------------|----|
|                           | 1.1    | Introd  | uction                                        |    |
|                           |        | 1.1.1   | Background                                    |    |
|                           |        | 1.1.2   | Objective                                     |    |
|                           |        | 1.1.3   | Using SystemVerilog instead of VHDL           |    |
|                           | 1.2    | Object  | t-oriented style in VHDL                      |    |
|                           |        | 1.2.1   | Understanding registers                       |    |
|                           |        | 1.2.2   | Inferring registers with variables            |    |
|                           |        | 1.2.3   | Creating instances                            |    |
|                           |        | 1.2.4   | Final OOP model                               |    |
|                           | 1.3    | Examp   | oles                                          | 1  |
|                           |        | 1.3.1   | Instances in series                           | 1  |
|                           |        | 1.3.2   | Instances in parallel                         | 1  |
|                           |        | 1.3.3   | Parallel instances in different clock domains | 1  |
|                           | 1.4    | Conclu  | usion                                         | 1  |
| $\mathbf{B}_{\mathbf{i}}$ | ibliog | raphy   |                                               | 1. |

## Chapter 1

# VHDL as intermediate language

This chapter develops synthesizable object-oriented (OOP) programming model for VHDL. Main motivation is to use it as an intermediate language for High-Level synthesis.

### 1.1 Introduction

### 1.1.1 Background

The most commonly used design 'style' for synthesizable VHDL models is what can be called the 'dataflow' style. A larger number of concurrent VHDL statements and small processes connected through signals are used to implement the desired functionality. Reading and understanding dataflow VHDL code is difficult since the concurrent statements and processes do not execute in the order they are written, but when any of their input signals change value. [1]

The biggest difference between a program in VHDL and standard programming language such as C, is that VHDL allows concurrent statements and processes that are scheduled for execution by events rather than in then order they are written. This reflects indeed the dataflow behaviour of real hardware, but is difficult to understand and analyse. On the other hand, analysing the behaviour of programs written in sequential programming languages does not become a problem even if the program tends to grow, since execution is done sequentially from top to bottom [1].

Jiri Gaisler has proposed an 'Structured VHDL design method [1]' in the ~2000 .He suggest to raise the hardware design abstraction level by using two-process method.

The two-process method only uses two processes per entity: one process that contains all combinatory (asynchronous) logic, and one process that contains all sequential logic (registers). Using this structure, the complete algorithm can be coded in sequential (non-concurrent) statements in the combinational process while the sequential process only contains registers, i.e. the state [1].

Object-oriented style in VHDL has been studied before. In [2] proposal was made to extend VHDL language with OOP semantics (dataflow based), this effort ended with development of OO-VHDL [3], an VHDL preprocessor, turning proposed extensions to standard VHDL. This work did not make it do VHDL standard, the status of compiler is unknown.

Many tools on the market are capable of convert higher level language to VHDL. However these tools only make use of the very basic dataflow semantics of VHDL language, resulting in complex conversion process and typically unreadable VHDL output.

The author of MyHDL package has written good blog posts about signal assignments [4] and software side of hardware design [5]. These ideas are relevant for this chapter.

### 1.1.2 Objective

Main motivation of this work is to use VHDL as an intermediate language for High-Level synthesis.

While the work of Jiri Gaisler greatly simplifies the programming experience of VHDL, it still has some major drawbacks:

- It is applicable only to single-clock designs [1]
- The 'structured' part can be only used to define combinatory logic, registers must be still inferred by signals assignments.
- It still relies on many of the VHDL dataflow features, for example, design reuse is achieved trough the use of entities and port maps.

This work aims to improve the 'two process' model by proposing Object-oriented way for VHDL, lifting all the previously listed drawbacks.

This sub-chapter uses examples in Python language in order to demonstrate the Python to VHDL converter (developed in next chapter) and set some targets for the intermediate language.

Multiply-accumulate(MAC) circuit is used as a demonstration circuit throughout the rest of this chapter.

Listing 1.1: Pipelined multiply-accumulate(MAC) implemented in Pyha

```
class MAC:
    def __init__(self, coef):
        self.coef = coef
        self.mul = 0
        self.acc = 0

def main(self, a):
        self.next.mul = a * self.coef
```

```
self.next.acc = self.acc + self.mul
return self.acc
```

**Note:** In order to keep examples simple, only integer types are used in this chapter.

Listing 1.1 shows a MAC component implemented in Pyha (Python to VHDL compiler implemented in the next chapter of this thesis) Operation of this circuit is to multiply the input with coefficient and accumulate the result. It synthesizes to logic shown on Fig. 1.1.



Fig. 1.1: Synthesis result of Listing 1.1 (Intel Quartus RTL viewer)

Main reason to pursue the OOP approach is the modularity and the ease of reuse. Listing 1.2 defines new class, containing two MACs that are to be connected in series. As expected it synthesizes to a series structure (Fig. 1.2).

Listing 1.2: Two MAC's connected in series

```
class SeriesMAC:
    def __init__(self, coef):
        self.mac0 = MAC(123)
        self.mac1 = MAC(321)

    def main(self, a):
        out0 = self.mac0.main(a)
        out1 = self.mac1.main(out0)
        return out1
```



Fig. 1.2: Synthesis result of Listing 1.2 (Intel Quartus RTL viewer)

With slight modification to the 'main' function (Listing 1.3), two MAC's can be connected in a way that synthesizes to parallel structure (Fig. 1.3).

1.1. Introduction 3

Listing 1.3: Two MAC's in parallel

```
def main(self, a):
    out0 = self.mac0.main(a)
    out1 = self.mac1.main(a)
    return out0, out1
```



Fig. 1.3: Synthesis result of Listing 1.3 (Intel Quartus RTL viewer)

It is clear that OOP style could significantly simplify hardware design. Objective of this work is to develop synthesizable VHDL model that could easily map to these MAC examples.

## 1.1.3 Using SystemVerilog instead of VHDL

SystemVerilog (SV) is the new standard for Verilog language, it adds significant amount of new features to the language [6]. Most of the added synthesizable features already existed in VHDL, making the synthesizable subset of these two languages almost equal. In that sense it is highly likely that ideas developed in this chapter could apply for both programming languages.

However in my opinion, SV is worse IR language compared to VHDL, because it is much more permissive. For example it allows out-of-bounds array indexing. This 'feature' is actually written into the language reference manual [7]. VHDL would error out the simulation, possibly saving debugging time.

While some communities have considered the verbosity and strictness of VHDL to be a downside, in my opinion it has always been an strength, and even more now when the idea is to use it as IR language.

Only motivation for using SystemVerilog over VHDL is tool support. For example Yosys [8], open-source synthesys tool, supports only Verilog, however to my knowledge it does not yet support SystemVerilog features. There have been also some efforts in adding VHDL frontend [9].

## 1.2 Object-oriented style in VHDL

While VHDL is mostly known as a dataflow language, it inherits strong support for structured programming from ADA.

Basic idea of OOP is to bundle up some common data and define functions that can perform actions on it. Then one could define multiple sets of the data. This idea fits well with hardware design, as 'data' can be thought as registers and combinatory logic as functions that perform operations on the data.

VHDL includes an 'class' like structure called 'protected types' [10], unfortunately these are not meant for synthesis. Even so, OOP style can be imitated, by combining data in records and passing it as a parameter to 'class functions'. This is essentially the same way how C programmers do it.

Listing 1.4: MAC data model in VHDL

```
type self_t is record
   mul: integer;
   acc: integer;
   coef: integer;
end record;
```

Constructing the data model for the MAC example can be done by using VHDL 'records' (Listing 1.4). In the sense of hardware, we expect that the contents of this record will be synthesised as registers.

**Note:** We label the data model as 'self', to be equivalent with the Python world.

Listing 1.5: OOP style function in VHDL (implementing MAC)

```
procedure main(self: inout self_t; a: in integer; ret_0: out integer) is
begin
    self.mul := a * self.coef;
    self.acc := self.acc + self.mul;
    ret_0 := self.acc;
end procedure;
```

OOP style function can be constructed by adding first argument, that points to the data model object (Listing 1.5). In VHDL procedure arguments must have a direction, for example the first argument 'self' is of direction 'inout', this means it can be read and also written to.

One drawback of VHDL procedures is that they cannot return a value, instead 'out' direction arguments must be used. Advantage of this is that the procedure may 'output/return' multiple values, as can Python functions.



Fig. 1.4: Synthesis result of Listing 1.5 (Intel Quartus RTL viewer)

Synthesis results (Fig. 1.4) show that functionally correct MAC has been implemented. However,in terms of hardware, it is not quite what was wanted. Data model specified 3 registers, but only the one for 'acc' is present and even this is at the wrong location.

In fact, the signal path from in0 to out0 contains no registers at all, making this design useless.

#### 1.2.1 Understanding registers

Clearly the way of defining registers is not working properly. Mistake was to expect that the registers work in the same way as 'class variables' in traditional programming languages.

In traditional programming, class variables are very similar to local variables. Difference is that class variables can 'remember' the value, while local variables exist only during the function execution.

Hardware registers have just one difference to class variables, value assigned to them does not take effect immediately, rather on the next clock edge. That is the basic idea of registers, they take new value on clock edge. When value is set at **this** clock edge, it will be taken on **next** clock edge.

Trying to stay in the software world, we can abstract away the clock edge by thinking that it denotes the call to the 'main' function. Meaning that registers take the assigned value on the next function call, meaning assignment is delayed by one function call.

VHDL defines a special assignment operator for this kind of delayed assignment, it is called 'signal assignment'. It must be used on VHDL signal objects like a <= b.

Jan Decaluwe, the author of MyHDL package, has written good article about the necessity of signal assignment semantics [4].

Using an signal assignment inside a clocked process always infers a register, because it exactly represents the register model.

### 1.2.2 Inferring registers with variables

While 'signals' and 'signal assignment' are the VHDL way of defining registers, it poses a major problem because they are hard to map to any other language than VHDL. This work aims to use variables instead, because they are the same in every other programming language.

VHDL signals really come down to just having two variables, to represent the **next** and **current** values. Signal assignment operator sets the value of **next** variable. On the next simulation delta, **current** is automatically set to equal **next**.

This two variable method has been used before, for example Pong P. Chu, author of one of the best VHDL books, suggests to use this style in defining sequential logic in VHDL [11]. Same semantics are also used in MyHDL [4].

Adapting this style for the OOP data model is shown on Listing 1.6.

Listing 1.6: Data model with **next** 

```
type next_t is record
   mul: integer;
   acc: integer;
   coef: integer;
end record;

type self_t is record
   mul: integer;
   acc: integer;
   coef: integer;
   coef: integer;
end record;
```

New data model allows reading the register value as before and extends the structure to include the 'nexts' object, so that it can used to assign new value for registers, for example self.nexts.acc := 0.

Integration of the new data model to the 'main' function is shown on Listing 1.7. Only changes are that all the 'register writes' go to the 'nexts' object.

Listing 1.7: Main function using 'nexts'

```
procedure main(self: inout self_t; a: integer; ret_0: out integer) is
begin
    self.nexts.mul := a * self.coef;
    self.nexts.acc := self.acc + self.mul;
    ret_0 := self.acc;
end procedure;
```

Last thing that must be handled is loading the **next** to **current**. As stated before, this is done automatically by VHDL for signal assignment, by using variables we have to take care of this ourselves. Listing 1.8 defines new function 'update\_registers', taking care of this task.

Listing 1.8: Function to update registers

```
procedure update_register(self: inout self_t) is
begin
    self.mul := self.nexts.mul;
    self.acc := self.nexts.acc;
    self.coef:= self.nexts.coef;
end procedure;
```

**Note:** Function 'update\_registers' is called on clock raising edge. It is possible to infer multi-clock systems by updating subset of registers at different clock edge.



Fig. 1.5: Synthesis result of the upgraded code (Intel Quartus RTL viewer)

Fig. 1.5 shows the synthesis result of the latest code. It is clear that this is now equal to the system presented at the start of this chapter.

## 1.2.3 Creating instances

General approach of creating instances is to define new variables of the 'self\_t' type, Listing 1.9 gives an example of this.

Listing 1.9: Class instances by defining records

```
variable mac0: MAC.self_t;
variable mac1: MAC.self_t;
```

Next step is to initialize the variables, this can be done at the variable definition, for example: variable mac0: self\_t := (mul=>0, acc=>0, coef=>123, nexts=>(mul=>0, acc=>0, coef=>123));

Problem with this method is that all data-model must be initialized (including 'nexts'), this will get unmaintainable very quickly, imagine having an instance that contains another instance or even array of instances. In some cases it may also be required to run some calculations in order to determine the initial values.

Traditional programming languages solve this problem by defining class constructor, executing automatically for new objects.

In the sense of hardware, this operation can be called 'reset' function. Listing 1.10 is a reset function for the MAC circuit. It sets the initial values for the data model and can also be used when reset signal is asserted.

Listing 1.10: Reset function for MAC

```
procedure reset(self: inout self_t) is
begin
    self.nexts.coef := 123;
    self.nexts.mul := 0;
    self.nexts.sum := 0;
    update_registers(self);
end procedure;
```

But now the problem is that we need to create new reset function for each instance.

This can be solved by using VHDL 'generic packages' and 'package instantiation declaration' semantics [10]. Package in VHDL just groups common declarations to one namespace.

In case of the MAC class, 'coef' reset value could be set as package generic. Then each new package initialization could define new reset value for it (Listing 1.11).

Listing 1.11: Initialize new package MAC\_0, with 'coef' 123

```
package MAC_0 is new MAC
generic map (COEF => 123);
```

Unfortunately, these advanced language features are not supported by most of the synthesis tools. Workaround is to either use explicit record initialization (as at the start of this chapter) or manually make new package for each instance.

Both of these solutions require unnecessary work load.

The Python to VHDL converter (developed in next chapter), uses the later option, it is not a problem as everything is automated.

#### 1.2.4 Final OOP model

Currently the OOP model consists of following elements:

- Record for 'next'
- Record for 'self'
- User defined functions (like 'main')
- 'Update registers' function

#### • 'Reset' function

VHDL supports 'packages' to group common types and functions into one namespace. Package in VHDL must contain an declaration and body (same concept as header and source files in C).

Listing 1.12 shows the template package for VHDL 'class'. All the class functionality is now in common namespace.

Listing 1.12: Package template for OOP style VHDL

```
package MAC is
    type next_t is record
    end record;
    type self_t is record
        nexts: next_t;
    end record;
    procedure reset(self: inout self_t);
    procedure update_registers(self: inout self_t);
    procedure main(self:inout self_t);
    -- other user defined functions
end package;
package body MAC is
    procedure reset(self: inout self_t) is
    begin
    end procedure;
    procedure update_registers(self: inout self_t) is
    begin
        . . .
    end procedure;
    procedure main(self:inout self_t) is
    begin
    end procedure;
    -- other user defined functions
end package body;
```

## 1.3 Examples

This chapter provides some simple examples based on the MAC component and OOP model, that were developed in previous chapter.

#### 1.3.1 Instances in series

Creating new class that connects two MAC instances in series is simple, first we need to create two MAC instances called MAC\_0 and MAC\_1 and add them to the data model (Listing 1.13).

Listing 1.13: Datamodel of 'series' class

```
type self_t is record
  mac0: MAC_0.self_t;
  mac1: MAC_1.self_t;

nexts: next_t;
end record;
```

Next step is to call MAC\_0 operation on the input and then pass the output trough MAC\_1, whom output is the final output (Listing 1.14).

Listing 1.14: Function that connects two MAC's in series

```
procedure main(self:inout self_t; a: integer; ret_0:out integer) is
   variable out_tmp: integer;
begin
   MAC_0.main(self.mac0, a, ret_0=>out_tmp);
   MAC_1.main(self.mac1, out_tmp, ret_0=>ret_0);
end procedure;
```



Fig. 1.6: Synthesis result of the new class (Intel Quartus RTL viewer)

Logic is synthesized in series (Fig. 1.6). That is exactly what was specified.

### 1.3.2 Instances in parallel

Connecting two MAC's in parallel can be done by just returning output of MAC\_0 and MAC\_1 (Listing 1.15).

1.3. Examples 11

Listing 1.15: Main function for parallel instances



Fig. 1.7: Synthesis result of Listing 1.15 (Intel Quartus RTL viewer)

Two MAC's are synthesized in parallel, as shown on Fig. 1.7.

#### 1.3.3 Parallel instances in different clock domains

Multiple clock domains can be easily supported by updating registers at specified clock domain. Listing 1.16 shows the contents of top-level process, where 'mac0' is updated by 'clk0' and 'mac1' by 'clk1'. Note that nothing has to be changed in the data model or main function.

Listing 1.16: Top-level for multiple clocks

```
if (not rst_n) then
    ReuseParallel_0.reset(self);
else
    if rising_edge(clk0) then
        MAC_0.update_registers(self.mac0);
    end if;

    if rising_edge(clk1) then
        MAC_1.update_registers(self.mac1);
    end if;
end if;
```

Synthesis result (Fig. 1.8) is as expected, MAC's are still in parallel but now the registers are clocked by different clocks. Reset signal is common for the whole design.



Fig. 1.8: Synthesis result with modified top-level process (Intel Quartus RTL viewer)

### 1.4 Conclusion

This chapter developed, synthesizable, object-oriented model for VHDL.

Major advantage is that none of the VHDL data-flow semantics are used (except for top level entity). This makes development similar to regular software. New programmers can learn this way much faster as the previous knowledge transfers.

Moreover, this model is not restricted to one clock domain and allows simple way of describing registers.

Major motivation for this model was to ease converting higher level languages into VHDL. This goal has been definitely reached, next section of this thesis develops Python bindings with relative ease. Conversion is drastically simplified as Python class maps to VHDL class, Python function maps to VHDL function and so on.

Synthesizability has been demonstrated using Intel Quartus toolset. Bigger designs, like frequency-shift-keying receiver, have been implemented on Intel Cyclone IV device. There has been no problems with hierarchy depth, objects may contain objects which itself may contain arrays of objects.

1.4. Conclusion 13

# Bibliography

- [1] Jiri Gaisler. A structured vhdl design method. URL: http://www.gaisler.com/doc/vhdl2proc.pdf.
- [2] Judith Benzakki and Bachir Djafri. Object Oriented Extensions to VHDL, The LaMI proposal, pages 334–347. Springer US, Boston, MA, 1997. URL: http://dx.doi.org/10.1007/978-0-387-35064-6\_27, doi:10.1007/978-0-387-35064-6\_27.
- [3] S. Swamy, A. Molin, and B. Covnot. Oo-vhdl. object-oriented extensions to vhdl. *Computer*, 28(10):18–26, Oct 1995. doi:10.1109/2.467587.
- [4] Jan Decaluwe. Why do we need signal assignments? URL: http://www.jandecaluwe.com/hdldesign/signal-assignments.html.
- [5] Jan Decaluwe. Thinking software at the rtl level. URL: http://www.jandecaluwe.com/hdldesign/thinking-software-rtl.html.
- [6] Stuart Sutherland and Don Mills. Synthesizing system verilog busting the myth that system verilog is only for verification. SNUG Silicon Valley, 2013.
- [7] Aurelian Ionel Munteanu. Gotcha: access an out of bounds index for a systemverilog fixed size array. URL: http://www.amiq.com/consulting/2016/01/26/gotcha-access-an-out-of-bounds-index-for-a-systemverilog-fixed-size-array/.
- [8] Clifford Wolf. Yosys open synthesis suite. URL: http://www.clifford.at/yosys/.
- [9] Florian Mayer. A vhdl frontend for the open-synthesis toolchain yosys. Master's thesis, Hochschule Rosenheim, 2016.
- [10] IEEE. Ieee standard vhdl language reference manual. IEEE Std 1076-2008 (Revision of IEEE Std 1076-2002), 2009.
- [11] Pong P. Chu. RTL Hardware Design Using VHDL: Coding for Efficiency, Portability, and Scalability. Wiley, 2006.