

# **Tutorial for Implementation of Quadruped Robot's Motion Control on SoC FPGA**

Karakasis Chrysostomos  
el13136@outlook.com

June 5, 2019

# Contents

|          |                                                              |           |
|----------|--------------------------------------------------------------|-----------|
| <b>1</b> | <b>Introduction</b>                                          | <b>2</b>  |
| <b>2</b> | <b>Installing Vivado and Digilent Board Files</b>            | <b>3</b>  |
| 2.1      | Install Minicom for SDK . . . . .                            | 3         |
| <b>3</b> | <b>Explanation of Vivado and VHDL Codes</b>                  | <b>4</b>  |
| 3.1      | Creating a new project . . . . .                             | 4         |
| 3.2      | Analysis of the Hardware Source Codes . . . . .              | 16        |
| 3.3      | Configuration of the Custom Made IP . . . . .                | 17        |
| 3.3.1    | <i>laelaps_four_legs_duplicate_ip_v1_0_S00_AXI</i> . . . . . | 17        |
| 3.3.2    | <i>laelaps_four_legs_duplicate_ip_v1_0</i> . . . . .         | 24        |
| 3.3.3    | Packaging the IP core . . . . .                              | 27        |
| 3.3.4    | Configuring the Block Design . . . . .                       | 34        |
| 3.3.5    | Constraints . . . . .                                        | 37        |
| 3.3.6    | Configuring the Operating Frequency of the FPGA . . . . .    | 38        |
| 3.3.7    | Generate Bitstream File . . . . .                            | 39        |
| <b>4</b> | <b>Software Partition in SDK</b>                             | <b>41</b> |
| 4.1      | Export the hardware design to SDK . . . . .                  | 41        |
| 4.2      | Preliminary Settings for the SDK . . . . .                   | 43        |
| 4.3      | Modify the Software Application . . . . .                    | 45        |
| 4.3.1    | Main Function . . . . .                                      | 45        |
| 4.3.2    | array_print function . . . . .                               | 46        |
| 4.3.3    | Interrupt Handler . . . . .                                  | 48        |
| 4.4      | Libraries Installation . . . . .                             | 49        |
| 4.5      | Load executables to the SD Card . . . . .                    | 50        |
| <b>5</b> | <b>Execution of the Implementation</b>                       | <b>54</b> |
| <b>6</b> | <b>Matlab Codes</b>                                          | <b>56</b> |

# Chapter 1

## Introduction

In this thesis, the Ubuntu 16.04.6 LTS Operating System was utilized, in addition to the following software tools:

- Vivado 2017.4, a software suite for the Hardware Design of the FPGA
- Xilinx Software Development Kit (XSDK) 2017.4, an Integrated Design Environment for the Software Design of the ARM processor
- Minicom, a program for the emulation of the SDK terminal
- Matlab, a computing environment for the visualization of the recorded measurements

In the next chapter, detailed guidelines are included, according to which all necessary programs can be installed and prepared for future use.

Consequently, each one of the four software tools will be analyzed, accompanied by explanatory instructions and step-by-step screenshots.

As depicted in the overall operational diagram (Fig.1.1), initially the user has to design the desired hardware architecture of the platform through Vivado. This includes both the programming of the FPGA, as well as the configuration of the communication between the ARM Processor and the FPGA. Next, via SDK, the user programs the processor and creates the appropriate file, which will be then transferred to the SD Card. Once the SD Card has been inserted to the Zybo Development Board, the execution of the previously configured hardware and software implementations begin, where the interfacing with the user is achieved via Minicom. After the termination of the experiment, all recorded measurements are stored in the SD Card, which can be extracted and inserted in a PC for post processing through Matlab.



Figure 1.1: Overall Operational Diagram.

# Chapter 2

## Installing Vivado and Digilent Board Files

In this chapter, an available on line tutorial is quoted from the website of Digilent Company. The actual link is given here<sup>\*</sup>, while the corresponding pdf file is also attached along with this report<sup>†</sup>, in case the web address is modified in the future. Apart for the installation of the Vivado Design Suite, the Digilent Board Files have to be installed, as they include the corresponding file for the currently employed Zybo Development Board.

Although the cited tutorial refers to the latest version (2018.3), the same procedure can be followed for any version. In our case, the Vivado 2017.4 Design Suite version was installed for Linux and it is also recommended for future users. If the user wishes to download a more updated version, the included source codes may present compatibility errors.

### 2.1 Install Minicom for SDK

In case the user selects to operate in a Windows Operating System, the Minicom is not necessary. However, due to incompatibility issues with Linux, in order to control and monitor the SDK terminal, Minicom has to be installed. A detailed installation guide can be found here<sup>‡</sup>, while the corresponding pdf file is also attached along with this report<sup>§</sup>, in case the web address is modified in the future. The same settings were applied in the context of this thesis, as instructed in the given installation guide.

---

<sup>\*</sup><https://reference.digilentinc.com/vivado/installing-vivado/start>

<sup>†</sup>Installing Vivado and Digilent Board Files.pdf

<sup>‡</sup>[http://processors.wiki.ti.com/index.php/Setting\\_up\\_Minicom\\_in\\_Ubuntu](http://processors.wiki.ti.com/index.php/Setting_up_Minicom_in_Ubuntu)

<sup>§</sup>Setting up Minicom in Ubuntu - Texas Instruments Wiki.pdf

# Chapter 3

## Explanation of Vivado and VHDL Codes

You are highly recommended to experiment with Vivado and VHDL language through several tutorials that exist online by Xilinx, such as *The Zynq Book Tutorials* and *Vivado Design Suite User Guide - Getting Started*, which are included in the **diploma\_thesis\_codes** folder.

### 3.1 Creating a new project

Once the installation of the Vivado Design Suite has been completed, we can open the Vivado 2017.4 application and start the design of our implementation. The main start window of Vivado is depicted in Fig.3.1, where we can create a new project or open a recent one.



Figure 3.1: The main start window of Vivado.

First we select to create a new project and press "Next" (Fig.3.2).



Figure 3.2: Create a new project.

Next, we have to choose a name and specify the location of the new project. In this case, we will name the project "laelaps\_four\_legs\_tutorial" (Fig.3.2).



Figure 3.3: Name and location of the new project.

Most of the times, the type of project to create will be "RTL Project", unless the designer wishes to select another option, for example to modify an already created project (Fig.3.4).



Figure 3.4: Selection of Project Type.

Consequently, in the next window pane (Fig.3.5), we have the ability to upload existing source files such as VHDL codes, IP files, etc. At this point we shall not utilize some existing codes, although this feature is quite useful, when we want to reuse some source codes.



Figure 3.5: Add Sources.

In the "Add Constraints" window (Fig.3.6), we will upload a constraints file, which holds important information regarding the physical ports of the Zybo Development Board such as switches, buttons, LEDS and of course the Pmods. Specifically, through this file it is possible to modify the corresponding name of each port and hence refer to it and utilize it in our design. The file is named "Zybo-Master.xdc" and it is located within the "diploma\_thesis\_codes.zip" file. Namely, the actual path is:

"diploma\_thesis\_codes/simple\_hello\_world/simple\_hello\_world.srcts/constrs\_1/imports/src". Once it has been selected, press "OK" and then "Next".



Figure 3.6: Add Constraints.



Figure 3.7: Location and Name of the Constraints file.

A crucial step for the creation of a new project is the determination of the board device on which we intend to execute our design (Fig.3.8). In the context of this thesis the Zynq Developement Board (Zybo) was utilized. First, we select "Boards" and then search for the term "zybo". Once the appropriate Display Name is available, we select it and press "Next". If the "Zybo" or "Zybo Z7-10" options are not available, there is probably something wrong with the installation of the Digilent Board Files, therefore refer to Chapter 2.



Figure 3.8: Selection of Default Part.

Finally, the settings for the new project are summarized in one window, where we can inspect them for any errors (Fig.3.9). If everything is in order we press "Finish" and otherwise we return to the previous stages via the "Back Button".



Figure 3.9: New Project Summary.

After several seconds the Vivado Project Manager will open, where the programming of our implementation will take place (Fig.3.10).



Figure 3.10: Project Manager.

The first step of our implementation is the creation of the *laelaps\_four\_legs\_qei\_pwm\_1.0* Custom IP, which is responsible for the processing of the encoders' signals, the determination of their positions and velocities, the creation of PWM signals, as well as the communication with the ARM Processor via the AXI4-Lite Interface. In the context of this tutorial, we will create a duplicate of the Custom IP that was realized during the thesis. In order to do so, we select "Tools" and then click on the "Create and Package New IP..." option (Fig.3.11).



Figure 3.11: Create and Package New IP.

This will open up a new window (Fig.3.12), which informs us about the abilities of this feature. We select "Next" and move on to the Fig.3.13, where we select the "Create a new AXI4 peripheral" option and click "Next".



Figure 3.12: Create and Package New IP First Window.



Figure 3.13: Create and Package New IP Second Window.

In Fig.3.14) we define the name of the new custom IP (*laelaps\_four\_legs\_duplicate\_ip*) and specify the location, where it will be stored later. Normally, all IPs are stored inside the *ip\_repo* folder. If it has not been automatically created, it is strongly recommended to create it and store there all IPs, either custom made or downloaded. Once we have completed all fields, we click "Next".



Figure 3.14: New Custom IP Name.

In the "Add Interfaces" window (Fig.3.15), we determine the characteristics of the AXI4 Interface that our Custom IP will support. In this case, an AXI4-Lite Interface will be created, due to the fact that the size of data to be transmitted is rather low. Additionally, for reasons of simplicity, our implementation dedicates three 32-bit slave registers for each joint, one for the motor's velocity, one for the encoder's position and the velocity's sign and one for the corresponding PWM signal. As a result, for four legs and hence 8 joints, 24 32-bit slave registers are required. Once we have completed all fields, we click "Next".



Figure 3.15: New Custom IP - Add Interfaces.

In the last "Create and Package New IP" window (Fig.3.16), our choices are summarized and we select the next step of our design. Since we will modify the IP and insert our design, the "Edit IP" bullet is selected and then we click "Finish".



Figure 3.16: New Custom IP - Summary.

After a few seconds, a new Vivado project, named *edit\_laelaps\_four\_legs\_duplicate\_ip\_v1\_0*, will open (Fig.3.17), through which the programming of the Custom IP can be achieved.



Figure 3.17: Project Manager for New Custom IP.

In the *Sources* pane (Fig.3.18), you should see two HDL source files:



Figure 3.18: Source Pane of New IP.

As we specified our target language as VHDL in Fig.3.5 earlier, the template files have been generated in VHDL. Had we specified Verilog as the target language, Verilog source files would have been created. The two source files are:

- **`labeled_four_legs_duplicate_ip_v1_0`** — This file instantiates all AXI-Lite interfaces. In this case, only one interface is present.
- **`labeled_four_legs_duplicate_ip_v1_0_S00_AXI.vhd`** — This file contains the AXI4-Lite interface functionality which handles the interactions between the peripheral in the PL and the software running on the PS.

The *IP Packager* pane will also be open in the Workspace:



Figure 3.19: IP Packager.

The information that we specified about our peripheral will be visible.

We can now add the functionality to our `labeled_four_legs_duplicate_ip` peripheral. We will be adding several source HDL files, which will be utilized as components in our peripheral.

In order to do that, we click the + symbol in the *Sources* pane.



Figure 3.20: Add Sources in the New Custom IP.

A new menu opens up, where we have to select whether we want to add or create constraints, design sources, and simulation sources. In this case, since we wish to add design sources, the second choice is selected and the "Next" button is clicked.



Figure 3.21: Add Sources in the New Custom IP - Second Figure.

Next, we click the "Add Files" button, which enables us to browse through our files and select which ones we want to integrate to our design. Specifically, the following source files will be selected, which can be found here: *diploma\_thesis\_codes/laelaps\_four\_legs\_qei\_vel\_pwm\_1.0/src*.

- *encoder.vhd*, responsible for the position calculation using the encoders' signals
- *filter.vhd*, responsible for the filtering of the encoders' outputs
- *pwm\_freq.vhd*, responsible for the creation of PWM signals
- *vel\_qei.vhd*, responsible for the velocity estimation
- *qei.vhd*, the top module which integrates and synchronizes the *encoder*, *filter*, *vel\_qei* components

If we wanted to create a new source file, we would choose the "Create File" option, which enables us to determine the file's characteristics such as its name and type.

Once all source files have been selected (Fig.3.22), we click "Finish" and return back to Project Manager window.



Figure 3.22: Add Sources in the New Custom IP - Add Files.

Now the following state should be visible (Fig.3.23), where the *qei* and *pwm\_freq* units have been included. As you can observe, the *filter*, *encoder* and *vel\_qei* units are integrated inside the *qei* block. This feature will be discussed promptly, as we explain the functionality of each unit.



Figure 3.23: Add Sources in the New Custom IP - Third.

## 3.2 Analysis of the Hardware Source Codes

As we mentioned above, the purpose of the Custom IP is to manage the calculation of the position and the velocity for each motor, the creation of PWM signals, as well as the communication with the ARM Processor. The first three tasks are been realized via corresponding hardware source codes written in VHDL.

First of all, the **filter** code is responsible for the stabilization of the system's inputs \*. In particular, the block receives as input a 1-bit signal and if it is stable then the block exports the signal as output. By using a fast clock, the system takes three samples of the input and if all three samples are in accordance with each other, the signal is considered stable. In this application the input signal is one of the three signals generated from the encoders (A, B, Index). Secondly, the **encoder** code emulates a Quadrature Encoder Interface (QEI) as described in the Diploma Thesis †. Thirdly, the **vel\_qei** code realizes the velocity estimation unit of our implementation, as described in the Diploma Thesis ‡ .

Now, the **qei** block is responsible for the connection, synchronization and management of the previous three units, which it utilizes as components. Initially, it creates a four times slower clock signal than the one it operates with. Therefore, we have two clock signals, one fast and one slow. Hereupon, it receives as inputs the unfiltered signals generated from the encoder (A, B and Index) and supplies them as inputs to three **filter** components. Each one of them operates according to the fast clock signal, whose frequency will be specified later, and outputs the stable and filtered encoder's signal. Consequently, the filtered signals are supplied as inputs to the **encoder** unit, along with the slow clock signal, while its generic variables are being defined at this point. It is important to note that the values assigned here bypass any other declaration inside the **encoder** code. It is quite similar to calling a C function and specifying its arguments. Then the **encoder** block outputs the motor's direction and position signals, which are being forwarded as inputs to the **vel\_qei** block, along with the slow clock signal. Additionally, the position signal is linked with one of the **qei** block's outputs. Finally, the velocity's value and sign are being computed and exported as outputs of the **qei** block, along with the aforementioned position signal.

Now, due to the fact that the **qei** block utilizes three **filter**, one **encoder** and one **vel\_qei** components, we can see those instances entailed under the **qei** unit in the *Sources Pane* in Fig.3.23.

Regarding the **pwm\_freq** code, as described in the Diploma Thesis §, it is in charge of the creation of PWM signals for the control of the power supplied to the quadruped's motors.

In total, the **qei** and **pwm\_freq** are the top modules that will also be utilized as components for the **laelaps\_four\_legs\_duplicate\_ip\_v1\_0\_S00\_AXI** entity, which will additionally configure the communication between the FPGA and the ARM Processor.

All corresponding source codes along with helpful comments can be found inside the **laelaps\_four\_legs\_duplicate\_ip** IP through this link:

[diploma\\_thesis\\_codes/laelaps\\_four\\_legs\\_duplicate\\_ip\\_1.0/src](diploma_thesis_codes/laelaps_four_legs_duplicate_ip_1.0/src)

---

\*Refer to Section 4.1 of the Diploma Thesis (Page 49)

†Refer to Section 4.2 of the Diploma Thesis (Page 50)

‡Refer to Section 4.3 of the Diploma Thesis (Page 52 and especially Page 54)

§Refer to Section 4.5 of the Diploma Thesis (Page 57)

### 3.3 Configuration of the Custom Made IP

### 3.3.1 laelaps\_four\_legs\_duplicate\_ip\_v1\_0\_S00\_AXI

For the configuration of the *laelaps\_four\_legs\_duplicate\_ip\_v1\_0\_S00\_AXI* source file, we basically modify the code in specific regions destined for the User, such as Lines 8 and 19 in Fig.3.24.

```
library ieee;
use ieee.std_logic_1164.all;
use ieee.numeric_std.all;
entity laelaps_four_legs_duplicate_ip_v1_0_S00_AXI is
    generic (
        -- Users to add parameters here
        -- User parameters ends
        -- Do not modify the parameters beyond this line
        -- Width of S_AXI data bus
        C_S_AXI_DATA_WIDTH : integer := 32;
        -- Width of S_AXI address bus
        C_S_AXI_ADDR_WIDTH : integer := 7
    );
    port (
        -- Users to add ports here
        -- User ports ends
        -- Do not modify the ports beyond this line
        -- Global Clock Signal
        S_AXI_ACLK : in std_logic;
        -- Global Reset Signal. This Signal is Active LOW
        S_AXI_ARESETN : in std_logic;
        -- Write address (issued by master, accepted by Slave)
        S_AXI_WADDR : in std_logic_vector(C_S_AXI_ADDR_WIDTH-1 downto 0);
        -- Write channel Protection type. This signal indicates the
        -- privilege and security level of the transaction, and whether
        -- the transaction is a data access or an instruction access.
        S_AXI_WPROT : in std_logic_vector(2 downto 0);
        -- Write address valid. This signal indicates that the master signaling
        -- the write address and protection information.
        S_AXI_WVALID : in std_logic;
        -- Write response (issued by Slave, accepted by master)
        S_AXI_WRESP : out std_logic;
        -- Write data (issued by master, accepted by Slave)
        S_AXI_WDATA : in std_logic_vector(C_S_AXI_DATA_WIDTH-1 downto 0);
        -- Write strobe. This signal indicates that the master is
        -- asserting the write data.
        S_AXI_WSTRB : in std_logic_vector(C_S_AXI_DATA_WIDTH/8-1 downto 0);
        -- Write last. This signal indicates that there is no
        -- subsequent write operation following this one.
        S_AXI_WLAST : in std_logic;
        -- Read address (issued by master, accepted by Slave)
        S_AXI_ARADDR : in std_logic_vector(C_S_AXI_ADDR_WIDTH-1 downto 0);
        -- Read channel Protection type. This signal indicates the
        -- privilege and security level of the transaction, and whether
        -- the transaction is a data access or an instruction access.
        S_AXI_ARPROT : in std_logic_vector(2 downto 0);
        -- Read address valid. This signal indicates that the master signaling
        -- the read address and protection information.
        S_AXI_ARVALID : in std_logic;
        -- Read response (issued by Slave, accepted by master)
        S_AXI_RRESP : out std_logic;
        -- Read data (issued by Slave, accepted by master)
        S_AXI_RDATA : out std_logic_vector(C_S_AXI_DATA_WIDTH-1 downto 0);
        -- Read strobe. This signal indicates that the slave is
        -- asserting the read data.
        S_AXI_RSTRB : out std_logic;
        -- Read last. This signal indicates that there is no
        -- subsequent read operation following this one.
        S_AXI_RLAST : out std_logic;
        -- Error output
        S_AXI_RREADY : out std_logic
    );
end entity;
```

Figure 3.24: Regions where the User is allowed to insert new code.

Initially, we specify the generic variables of our IP (Fig.3.25) and add the necessary ports (Fig.3.26).

Figure 3.25: Adding the necessary Generic Variables.

The ports correspond to the Pmod Ports through which the reading of the encoders' signals and the transmission of the PWM signals will be achieved. The users can name those ports anyway they desire, but in this case we named them after some specific keywords that existed in the constraints file we added earlier (Fig.3.7) that will be elaborated in a later section (3.3.5). In addition, two sets of ports are included that represent the Zybo Development Board's switches and LEDs.



```

laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd *
/home/chris/ip_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd

25 port (
26   -- Users to add ports here
27   jb_p0 : in std_logic; --A1 of first leg
28   jb_n0 : in std_logic; --B1 of first leg
29   jb_p1 : in std_logic; --A2 of first leg
30   jb_n1 : in std_logic; --B2 of first leg
31   jb_p2 : out std_logic; --DIR1 of first leg
32   jb_n2 : out std_logic; --PWM1 of first leg
33   jb_p3 : out std_logic; --DIR2 of first leg
34   jb_n3 : out std_logic; --PWM2 of first leg
35
36   jc_p0 : in std_logic; --A1 of second leg
37   jc_n0 : in std_logic; --B1 of second leg
38   jc_p1 : in std_logic; --A2 of second leg
39   jc_n1 : in std_logic; --B2 of second leg
40   jc_p2 : out std_logic; --DIR1 of second leg
41   jc_n2 : out std_logic; --PWM1 of second leg
42   jc_p3 : out std_logic; --DIR2 of second leg
43   jc_n3 : out std_logic; --PWM2 of second leg
44
45   jd_p0 : in std_logic; --A1 of third leg
46   jd_n0 : in std_logic; --B1 of third leg
47   jd_p1 : in std_logic; --A2 of third leg
48   jd_n1 : in std_logic; --B2 of third leg
49   jd_p2 : out std_logic; --DIR1 of third leg
50   jd_n2 : out std_logic; --DIR2 of third leg
51   jd_p3 : out std_logic; --DIR2 of third leg
52   jd_n3 : out std_logic; --DIR2 of third leg
53
54   je_p0 : in std_logic; --A1 of fourth leg
55   je_n0 : in std_logic; --B1 of fourth leg
56   je_p1 : in std_logic; --A2 of fourth leg
57   je_n1 : in std_logic; --B2 of fourth leg
58   je_p2 : out std_logic; --DIR1 of fourth leg
59   je_n2 : out std_logic; --DIR2 of fourth leg
60   je_p3 : out std_logic; --DIR2 of fourth leg
61   je_n3 : out std_logic; --DIR2 of fourth leg
62   led : out std_logic_vector(3 downto 0); --leds of Zybo Board
63   sw : in std_logic_vector(3 downto 0); --switches of Zybo Board
64   -- User ports ends
65   -- Do not modify the ports beyond this line

```

Figure 3.26: Adding the necessary User Ports.

Afterwards, we specify any signals we want to utilize in our design, between the Lines 151-153 (Fig.3.27).



```

laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd *
/home/chris/ip_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd

145   -- local parameter for addressing 32 bit / 64 bit C_S_AXI_DATA_WIDTH
146   -- ADDR_LSB is used for addressing 32/64 bit registers/memories
147   -- ADDR_LSB = 2 for 32 bits (n downto 2)
148   -- ADDR_LSB = 3 for 64 bits (n downto 3)
149   constant ADDR_LSB : integer := (C_S_AXI_DATA_WIDTH/32)+ 1;
150   constant OPT_MEM_ADDR_BITS : integer := 4;
151
152   ---- Signals for user logic register space example
153   -----
154   ---- Number of Slave Registers 24
155   signal slv_reg0 :std_logic_vector(C_S_AXI_DATA_WIDTH-1 downto 0);
156   signal slv_reg1 :std_logic_vector(C_S_AXI_DATA_WIDTH-1 downto 0);
157   signal slv_reg2 :std_logic_vector(C_S_AXI_DATA_WIDTH-1 downto 0);
158   signal slv_reg3 :std_logic_vector(C_S_AXI_DATA_WIDTH-1 downto 0);
159   signal slv_reg4 :std_logic_vector(C_S_AXI_DATA_WIDTH-1 downto 0);

```

Figure 3.27: Adding the necessary User Signals.

Specifically, for each leg we define two sets of three signals for each joint (one for the position, one for the velocity's value and one for its sign) and two signals for the PWM of each joint's motor (Fig.3.28). In total eight signals for each leg.

```

labeled_four_legs_duplicate_ip_v1_0_S00_AXI.vhd
/home/chris/ip_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd

154: --LEG NUMBER 1
155: ...
156: ...
157: ...
158: ...
159: --signals for ENCODER 1 LEG 1
160: signal Counts1_1 : STD_LOGIC_VECTOR (bits-1 downto 0); --Counts of encoder 1 of leg 1
161: signal vel_sign1_1 : std_logic;
162: signal vell1_1 : std_logic_vector(31 downto 0); --Velocity of motor 1 of leg 1
163: ...
164: ...
165: ...
166: ...
167: ...
168: ...
169: ...
170: ...
171: ...

labeled_four_legs_duplicate_ip_v1_0_S00_AXI.vhd
/home/chris/ip_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd

172: --LEG NUMBER 2
173: ...
174: ...
175: ...
176: ...
177: ...
178: --signals for ENCODER 1 LEG 2
179: signal Counts1_2 : STD_LOGIC_VECTOR (bits-1 downto 0); --Counts of encoder 1 of leg 2
180: signal vel_sign1_2 : std_logic;
181: signal vell1_2 : std_logic_vector(31 downto 0); --Velocity of motor 1 of leg 2
182: ...
183: ...
184: ...
185: ...
186: ...
187: ...
188: ...
189: ...
190: ...

labeled_four_legs_duplicate_ip_v1_0_S00_AXI.vhd
/home/chris/ip_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd

191: --LEG NUMBER 3
192: ...
193: ...
194: ...
195: ...
196: ...
197: ...
198: ...
199: ...
200: ...
201: ...
202: ...
203: ...
204: ...
205: ...
206: ...
207: ...
208: ...
209: ...

labeled_four_legs_duplicate_ip_v1_0_S00_AXI.vhd
/home/chris/ip_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd

209: --LEG NUMBER 4
210: ...
211: ...
212: ...
213: ...
214: ...
215: ...
216: ...
217: ...
218: ...
219: ...
220: ...
221: ...
222: ...
223: ...
224: ...
225: ...
226: ...
227: ...
228: ...
229: ...
230: ...
231: ...
232: ...
233: ...
234: ...
235: ...
236: ...
237: ...
238: ...
239: ...
240: ...
241: ...
242: ...
243: ...
244: ...
245: ...
246: ...
247: ...
248: ...

```

Figure 3.28: User Signals for the Fours Legs

Moreover, some auxiliary signals are defined, which will serve as temporary variables (Fig.3.29).

```

labeled_four_legs_duplicate_ip_v1_0_S00_AXI.vhd
/home/chris/ip_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd

227: ...
228: ...
229: ...
230: ...
231: ...
232: ...
233: ...
234: ...
235: ...
236: ...
237: ...
238: ...
239: ...
240: ...
241: ...
242: ...
243: ...
244: ...
245: ...
246: ...
247: ...
248: ...

```

Figure 3.29: User Auxiliary Signals.

Subsequently, we declare the components that we plan on utilizing as they are have been defined in the corresponding source files (Fig.3.30). In this case, we will include the *qei* and *pwm\_freq* files, which were explained previously.

```

labeled_four_legs_duplicate_ip_v1_0_S00_AXI.vhd *
/home/chris/ip_repo/labeled_four_legs_duplicate_ip_1_0/hdl/labeled_four_legs_duplicate_ip_v1_0_S00_AXI.vhd

248 -----
249   --Declaration of components
250   component qei
251     generic(--these are not the actual values of the generic variables. See Generic Map Lines 486 & 502
252       sys_clk : INTEGER := 100_000_000;           --System clock frequency in Hz
253       bits : INTEGER := 18;                      --number of ratio bits
254       ratio_numerator : INTEGER := 98;          --numerator of the gear ratio of the motor-gearhead
255       ratio_divisor : INTEGER := 1;              --divisor of the gear ratio of the motor-gearhead
256     );
257     Port (
258       QEA : in STD_LOGIC;                         --The A signal as generated from the encoder
259       Index : in STD_LOGIC;                        --The Index signal as generated from the encoder
260       QEB : in STD_LOGIC;                         --The B signal as generated from the encoder
261       Clk : in STD_LOGIC;                         --The Clock Signal
262       Count_out : out STD_LOGIC_VECTOR(bits-1 downto 0); --The Calculated Position of the Motor in Counts
263       vel_sign : out STD_LOGIC;                   --The Calculated Sign of the Motor's Velocity
264       velocity : out STD_LOGIC_VECTOR (31 downto 0) --The Calculated Value of the Motor's Velocity
265     );
266   end component;
267
268   component pwm_freq IS
269     GENERIC(
270       --The values defined below will only apply in case the "pwm_freq" entity is used as the top module
271       --Otherwise, if another project utilizes it as a component, then the generic variables will be
272       --utilized by the component's own generic map
273       sys_clk : INTEGER := 100_000_000;           --System clock frequency in Hz
274       pwm_frequency : INTEGER := 20_000;          --PWM switching frequency in Hz
275       bits_resolution : INTEGER := 14;           --bits of resolution setting the duty cycle
276       phases : INTEGER := 1;                     --number of output pms and phases
277     PORT(
278       clk : IN STD_LOGIC;                       --system clock
279       reset : IN STD_LOGIC;                      --asynchronous reset
280       ena : IN STD_LOGIC;                        --latches in new duty cycle
281       duty : IN STD_LOGIC_VECTOR(bits_resolution-1 DOWNTO 0); --duty cycle
282       pwm_out : OUT STD_LOGIC_VECTOR(phases-1 DOWNTO 0)); --pwm outputs
283     END component;
284

```

Figure 3.30: Declaration of Components.

A large part of the remaining source code must remain unmodified, as it is vital to the internal functionality of the AXI4-Lite Interface. In fact, the next modification occurs after Line 737, where we either supply the slave registers with the signals we wish to send to the ARM Processor, or read data from them that were transmitted there by the Processor. However, the values of those signals will be set later in the code, together with the utilization of the slave registers designated for reading.

In order to explain how the slave registers work, we will examine the signals and registers assigned for the management of the first leg, since the same logic applies for the those destined for the rest of the legs (Fig.3.31).

As we had configured our IP in the beginning, 24 32-bit slave registers exist in total, each recognized with a corresponding binary expression. The "0000" code represents the first slave register, the "0001" code represents the second one, and so on.

As a result, in lines 750-752 the first slave register is configured, which in this case will be a sending register and transmit data from the FPGA to the ARM Processor. The first *bits* bits are set with the Position in counts of the first encoder of the first leg, where the generic variable *bits* is equal to 18 in our implementation. The 19th bit is set with the sign of the velocity for the first leg's first motor, while the remaining bits are not utilized and hence set to zero.

The second slave register ("0001") is also a sending register and is set with the value of the velocity for the first leg's first motor. As a remainder, in our implementation the velocity signal is a 32-bit signal and hence we have to employ all the bits of this slave register. This is the reason why we chose to send the velocity's sign through the previous slave register.

```

1 laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd *
2 /home/chris/ip_repo/laelaps_four_legs_duplicate_ip_v1_0/hdl/laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd
3
4 Q | H | ← | → | X | D | F | // | B | Q |
5
6 736 :
7 737   -- Implement memory mapped register select and read logic generation
8 738   -- Slave register read enable is asserted when valid address is available
9 739   -- and the slave is ready to accept the read address.
10 740   slv_reg_rden <= axi_arready and S_AXI_ARVALID and (not axi_rvalid);
11 741
12 742   process (Counts1_1, vel_sign1_1, vel1_1, slv_reg2, Counts2_1, vel_sign2_1, vel2_1, slv_reg5, Counts1_2, vel_sign1_2, vel1_2
13 743   variable loc_addr : STD_logic_vector(OPT_MEM_ADDR_BITS downto 0);
14 744 begin
15 745   -- Address decoding for reading registers
16 746   loc_addr := axi_araddr(ADDR_LSB + OPT_MEM_ADDR_BITS downto ADDR_LSB);
17 747   case loc_addr is
18 748     when b"00000" =>
19 749       reg_data_out <= slv_reg0;
20 750       reg_data_out(31 downto bits+1) <= (others => '0');
21 751       reg_data_out(bits) <= vel_sign1_1;          --Send the Sign of velocity for motor 1 leg 1
22 752       reg_data_out(bits-1 downto 0) <= Counts1_1;    --Send the Counter of QEI for encoder 1 leg 1
23 753     when b"00001" =>
24 754       reg_data_out <= slv_reg1;
25 755       reg_data_out(31 downto 0) <= vel1_1;           --Send the Velocity of motor 1 leg 1
26 756     when b"00010" =>
27 757       reg_data_out <= slv_reg2;                   --Read the register relevant with the PWM of motor 1 leg 1
28 758     when b"00011" =>
29 759       reg_data_out <= slv_reg3;
30 760       reg_data_out(31 downto bits+1) <= (others => '0');
31 761       reg_data_out(bits) <= vel_sign2_1;          --Send the Sign of velocity for motor 2 leg 1
32 762       reg_data_out(bits-1 downto 0) <= Counts2_1;    --Send the Counter of QEI for encoder 2 leg 1
33 763     when b"00100" =>
34 764       reg_data_out <= slv_reg4;
35 765       reg_data_out(31 downto 0) <= vel2_1;           --Send the Velocity of motor 2 leg 1
36 766     when b"00101" =>
37 767       reg_data_out <= slv_reg5;                   --Read the register relevant with the PWM of motor 2 leg 1

```

Figure 3.31: Configuration of Slave Registers of AXI4-Lite Interface.

The third slave register ("0010") is a reading register, which transfers information assigned in the Processor's domain. Basically, this signal is relevant to the creation of the PWM signal for the first leg's first motor. Some of its bits will determine the enable and reset signals and the Duty Cycle of a *pwm\_freq* component, while one of its bits stipulates the Direction Signal of the Motor.

Likewise, the same logic applies for the next three slave registers, which are designated for the first leg's second motor, while the overall procedure is repeated for the remaining legs. In summary, the signals supplied to the sending registers will be computed via a fitting number of *qei* components, while the reading registers will determine the inputs of corresponding *pwm\_freq* components.

The call of the *qei* and *pwm\_freq* components occurs after Line 852, where the user logic can be added (Fig.3.32).

```

1 laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd *
2 /home/chris/ip_repo/laelaps_four_legs_duplicate_ip_v1_0/hdl/laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd
3
4 Q | H | ← | → | X | D | F | // | B | Q |
5
6 851 :
7 852   -- Add user logic here
8 853   -- User logic ends
9 854
10 855 end arch_imp;
11 856
12 857

```

Figure 3.32: Inclusion of the user logic.

Specifically, 8 *qei* components are deployed, one for each joint of the quadruped. Once their generic variables are configured inside the *generic map*, the previously defined signals, which correspond to the Pmod Ports, are supplied as the inputs QEA and QEB, while the Index input is set to zero and hence disabled. The *S\_AXI\_ACLK* is the clock signal of our system, whose frequency can be specified later. Eventually, the three outputs of each component is linked with the intermediate signals that feed the sending slave registers (Fig.3.33).

```

laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd
/home/chris/p_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd

851      -- Add user logic here
852      -QEI's start here
853      c11: qei
854          generic map(sys_clk => sys_clk,
855                      bits => bits,
856                      ratio_numerator => hip_ratio_numerator,
857                      ratio_divisor => hip_ratio_divisor
858                      )
859          port map(
860              QEA => jb_p0,
861              Index => jb_0,
862              QEB => jb_n0,
863              CkL => S_AXI_ACLK,
864              Count_out => Counts1_1,
865              vel_sign => vel_sign1_1,
866              velocity => vel1_1
867          );

```

  

```

laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd
/home/chris/p_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd

965      c18: qei
966          generic map(sys_clk => sys_clk,
967                      bits => bits,
968                      ratio_numerator => knee_ratio_numerator,
969                      ratio_divisor => knee_ratio_divisor
970                      )
971          port map(
972              QEA => je_p1,
973              Index => "0",
974              QEB => je_n1,
975              CkL => S_AXI_ACLK,
976              Count_out => Counts2_4,
977              vel_sign => vel_sign2_4,
978              velocity => vel2_4
979          );
980      -QEI's end here
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002

```

Figure 3.33: The first and last *qei* components

Ultimately, 8 *pwm\_freq* components are employed, each responsible for the creation of each motor's corresponding PWM signal. At first, their generic variables are configured, while they receive as inputs the fitting bits of the reading slave registers that we discussed above. As outputs, we have the PWM signals, which together with the the third bits of the respective reading slave registers (Direction Signals) are redirected to the Pmod ports of the system that serve as outputs and will connect to the motors' drivers (Fig.3.34).

```

laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd
/home/chris/p_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd

984      -PWMs start here
985      -LEG NUMBER 1
986      p1: pwm_freq
987          generic map (sys_clk => sys_clk,
988                          pwm_frequency => pwm_frequency,
989                          bits_resolution => bits_resolution,
990                          phases => phases
991          )
992          port map(
993              clk => S_AXI_ACLK,
994              reset => slv_reg2(0),
995              ena => slv_reg2(1),
996              duty => slv_reg2(bits_resolution-1+3 downto 3),
997              pwm_out => pwm_output1_1
998          );
999
1000      jb_p2 <= slv_reg2(2);
1001      jb_n2 <= pwm_output1_1(0);
1002

```

  

```

laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd
/home/chris/p_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0_S00_AXI.vhd

1108      p2: pwm_freq
1109          generic map (sys_clk => sys_clk,
1110                          pwm_frequency => pwm_frequency,
1111                          bits_resolution => bits_resolution,
1112                          phases => phases
1113          )
1114          port map(
1115              clk => S_AXI_ACLK,
1116              reset => slv_reg23(0),
1117              ena => slv_reg23(1),
1118              duty => slv_reg23(bits_resolution-1+3 downto 3),
1119              pwm_out => pwm_output2_4
1120          );
1121
1122      je_p3 <= slv_reg23(2);
1123      je_n3 <= pwm_output2_4(0);
1124
1125      -PWMs end here

```

Figure 3.34: The first and last *pwm\_freq* components

The last modifications inside the *laelaps\_four\_legs\_duplicate\_ip\_v1\_0\_S00\_AXI* source file will be now elaborated, although they are not vital to the functionality of the implementation. For supervision purposes, we chose to redirect the PWM and Direction signals to the LEDs. However, due to the fact that the Zybo Board is equipped with only four LEDs, a mechanism is required to select which signals we wish to see at a given time to the LEDs. Therefore, a simple multiplexer was constructed, which according to the switches state, controls which signals are displayed. If the first switch is on and the rest off, the LEDs display the PWM and Direction signals of the first leg. Likewise, if the second switch is on and the rest off, the LEDs display the PWM and Direction signals of the second leg, and so on. The first two LEDs show the PWM and Direction signals of the leg's first motor, while the last two represent the PWM and Direction signals of the leg's second motor (Fig.3.35).

```

labeled_four_legs_duplicate_ip_v1_0_S00_AXI.vhd
/home/chris/p_repo/labeled_four_legs_duplicate_ip_v1_0/hdl/labeled_four_legs_duplicate_ip_v1_0_S00_AXI.vhd

1126 led0_temp0 <= pwm_output1_1 and sv(phases-1 downto 0);
1127 led0_temp1 <= pwm_output1_2 and sv(phases downto 1);
1128 led0_temp2 <= pwm_output1_3 and sv(phases+1 downto 2);
1129 led0_temp3 <= pwm_output1_4 and sv(phases+2 downto 3);
1130
1131 led1_temp0 <= slv_reg2(2) and sw(0);
1132 led1_temp1 <= slv_reg2(2) and sw(1);
1133 led1_temp2 <= slv_reg2(2) and sw(2);
1134 led1_temp3 <= slv_reg2(2) and sw(3);
1135
1136 led2_temp0 <= pwm_output2_1 and sv(phases-1 downto 0);
1137 led2_temp1 <= pwm_output2_2 and sv(phases downto 1);
1138 led2_temp2 <= pwm_output2_3 and sv(phases+1 downto 2);
1139 led2_temp3 <= pwm_output2_4 and sv(phases+2 downto 3);
1140
1141 led3_temp0 <= slv_reg2(2) and sw(0);
1142 led3_temp1 <= slv_reg1(2) and sw(1);
1143 led3_temp2 <= slv_reg2(2) and sw(2);
1144 led3_temp3 <= slv_reg1(2) and sw(3);
1145
1146 led(phases-1 downto 0) <= led0_temp0 or led0_temp1 or led0_temp2 or led0_temp3;
1147 led(1) <= led1_temp0 or led1_temp1 or led1_temp2 or led1_temp3;
1148 led(phases+1 downto 2) <= (led2_temp0 or led2_temp1 or led2_temp2 or led2_temp3;
1149 led(3) <= led3_temp0 or led3_temp1 or led3_temp2 or led3_temp3;
1150
-- User logic ends
1151
1152 end arch_imp;
1153

```

Figure 3.35: Switches-LEDs Multiplexer.

At this point, we save our changes in this source file and return to the *Sources* Pane, where we can observe that the *qei* and *pwm\_freq* units are not independent, but rather they are now located under the *labeled\_four\_legs\_duplicate\_ip\_v1\_0\_S00\_AXI* project. In fact, eight *qei* and eight *pwm\_freq* components are depicted, in correspondence to our changes in this source file (Fig.3.36).



Figure 3.36: Updated Source Pane.

### 3.3.2 laelaps\_four\_legs\_duplicate\_ip\_v1\_0

Last but no least, this source file serves as the top module our custom IP, which instantiates the AXI4-Lite Interface and defines all generic variables we have utilized so far.

Similarly to the previous source file, first we include this module's generic variables (Fig.3.37) and ports (Fig.3.38), which are exactly the same with the previous source file. However, since this is the top module, the values set in the generic variables now will broadcast to all the components we have described. Therefore, by changing the values here, we can control the generic variables in the whole implementation.

The system's clock frequency is set to 83333333 Hz, due to the fact that this frequency was found suitable to our design for reasons that will be explained later. The motors currently in use in Laelaps II and their drivers operate with PWM signals around the frequency of 20kHz and hence it is an acceptable value. The bits of resolution for the Duty Cycle were proved to be sufficient, although higher values could also be used. Since our PWM signals are required to have only one phase, the *phases* variable is set to "1". The 18 bits of precision for the encoders' counts were chosen in order for our system to support the current gear ratios of the motors, as explained in section 4.2 of the Diploma Thesis. On the other hand, the last four generic variables correspond to the current gear ratios, whose values are mentioned in subsection 3.4.4 of the Diploma Thesis. As a result, if the motors change in the future, the only thing that is required to be modified is these four variables.

The screenshot shows a VHDL code editor with the following details:

- Title Bar:** laelaps\_four\_legs\_duplicate\_ip\_v1\_0.vhd
- Status Bar:** /home/chris/p\_repo/laelaps\_four\_legs\_duplicate\_ip\_1.0/hdl/laelaps\_four\_legs\_duplicate\_ip\_v1\_0.vhd
- Toolbar:** Includes icons for search, file operations, and zoom.
- Code Area:** Displays the VHDL source code. The code defines an entity named "laelaps\_four\_legs\_duplicate\_ip\_v1\_0" with generic parameters. It includes comments explaining the generic parameters and ends with a note about not modifying them beyond a certain line. The code also defines a parameter block for the Axi Slave Bus Interface S00\_AXI.

```
library ieee;
use ieee.std_logic_1164.all;
use ieee.numeric_std.all;
entity laelaps_four_legs_duplicate_ip_v1_0 is
generic (
    -- This is the top module of our IP and hence the values defined here are the ones that determine the rest of the generic variables
    -- Users to add parameters here
    sys_clk : INTEGER := 83_333_333;          --system clock frequency in Hz
    pwm_frequency : INTEGER := 20_000;          --PWM switching frequency in Hz
    bits_resolution : INTEGER := 14;            --bits of resolution setting the duty cycle
    phases : INTEGER := 1;                      --number of phases
    bits : INTEGER := 18;                       --number of bits for encoder's counts
    hip_ratio_numerator : INTEGER := 1029;      --numerator of the gear ratio of the Hip motor-gearhead
    hip_ratio_divisor : INTEGER := 13;           --divisor of the gear ratio of the Hip motor-gearhead
    knee_ratio_numerator : INTEGER := 98;         --numerator of the gear ratio of the Knee motor-gearhead
    knee_ratio_divisor : INTEGER := 1;            --divisor of the gear ratio of the Hip motor-gearhead
    -- User parameters ends
    -- Do not modify the parameters beyond this line
    );
-- Parameters of Axi Slave Bus Interface S00_AXI
C_S00_AXI_DATA_WIDTH : integer := 32;
C_S00_AXI_ADDR_WIDTH : integer := 7
);
```

Figure 3.37: Top Module's Generic Variables.

Following that, we update the declaration of the *laelaps\_four\_legs\_duplicate\_ip\_v1\_0\_S00\_AXI* component, by adding its generic variables (Fig.3.39) and ports (Fig.3.40).

```

laelaps_four_legs_duplicate_ip_v1_0.vhd
/home/chris/p_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0.vhd

26 : -- Users to add ports here
27 : jb_p0 : in std_logic; --A1 of first leg
28 : jb_n0 : in std_logic; --B1 of first leg
29 : jb_p1 : in std_logic; --A2 of first leg
30 : jb_n1 : in std_logic; --B2 of first leg
31 : jb_p2 : out std_logic; --DIR1 of first leg
32 : jb_n2 : out std_logic; --PWM1 of first leg
33 : jb_p3 : out std_logic; --DIR2 of first leg
34 : jb_n3 : out std_logic; --PWM2 of first leg
35 :
36 : jc_p0 : in std_logic; --A1 of second leg
37 : jc_n0 : in std_logic; --B1 of second leg
38 : jc_p1 : in std_logic; --A2 of second leg
39 : jc_n1 : in std_logic; --B2 of second leg
40 : jc_p2 : out std_logic; --DIR1 of second leg
41 : jc_n2 : out std_logic; --PWM1 of second leg
42 : jc_p3 : out std_logic; --DIR2 of second leg
43 : jc_n3 : out std_logic; --PWM2 of second leg
44 :
45 : jd_p0 : in std_logic; --A1 of third leg
46 : jd_n0 : in std_logic; --B1 of third leg
47 : jd_p1 : in std_logic; --A2 of third leg
48 : jd_n1 : in std_logic; --B2 of third leg
49 : jd_p2 : out std_logic; --DIR1 of third leg
50 : jd_n2 : out std_logic; --PWM1 of third leg
51 : jd_p3 : out std_logic; --DIR2 of third leg
52 : jd_n3 : out std_logic; --PWM2 of third leg
53 :
54 : je_p0 : in std_logic; --A1 of fourth leg
55 : je_n0 : in std_logic; --B1 of fourth leg
56 : je_p1 : in std_logic; --A2 of fourth leg
57 : je_n1 : in std_logic; --B2 of fourth leg
58 : je_p2 : out std_logic; --DIR1 of fourth leg
59 : je_n2 : out std_logic; --PWM1 of fourth leg
60 : je_p3 : out std_logic; --DIR2 of fourth leg
61 : je_n3 : out std_logic; --PWM2 of fourth leg
62 : led : out std_logic_vector(3 downto 0); --leds of Zynq Board
63 : sw : in std_logic_vector(3 downto 0); --switches of Zynq Board
64 : -- User ports ends

```

Figure 3.38: Top Module's Ports.

```

laelaps_four_legs_duplicate_ip_v1_0.vhd
/home/chris/p_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0.vhd

92 : architecture arch_imp of laelaps_four_legs_duplicate_ip_v1_0 is
93 :   -- component declaration
94 :   component laelaps_four_legs_duplicate_ip_v1_0_S00_AXI is
95 :     generic (
96 :       sys_clk_frequency : INTEGER := 100_000_000; --system clock frequency in Hz
97 :       pwm_frequency : INTEGER := 20_000; --PWM switching frequency in Hz
98 :       bits_resolution : INTEGER := 14; --bits of resolution setting the duty cycle
99 :       phases : INTEGER := 1;
100 :       bits : INTEGER := 18; --number of bits for encoder's counts
101 :       hip_ratio_numerator : INTEGER := 1029; --numerator of the gear ratio of the Hip motor-gearhead
102 :       hip_ratio_divisor : INTEGER := 13; --divisor of the gear ratio of the Hip motor-gearhead
103 :       knee_ratio_numerator : INTEGER := 98; --numerator of the gear ratio of the Knee motor-gearhead
104 :       knee_ratio_divisor : INTEGER := 1; --divisor of the gear ratio of the Knee motor-gearhead
105 :       C_S_AXI_DATA_WIDTH : integer := 32; --divisor of the gear ratio of the Hip motor-gearhead
106 :       C_S_AXI_ADDR_WIDTH : integer := 7
107 :     );
108 :   end component;
109 : 
```

Figure 3.39: Top Module's Component Declaration - Generic Variables.

```

laelaps_four_legs_duplicate_ip_v1_0.vhd
/home/chris/p_repo/laelaps_four_legs_duplicate_ip_1.0/hdl/laelaps_four_legs_duplicate_ip_v1_0.vhd

108 : C_S_AXI_ADDR_WIDTH : integer := 7
109 :
110 : port (
111 :   jb_p0 : in std_logic; --A1 of first leg
112 :   jb_n0 : in std_logic; --B1 of first leg
113 :   jb_p1 : in std_logic; --A2 of first leg
114 :   jb_n1 : in std_logic; --B2 of first leg
115 :   jb_p2 : out std_logic; --DIR1 of first leg
116 :   jb_n2 : out std_logic; --PWM1 of first leg
117 :   jb_p3 : out std_logic; --DIR2 of first leg
118 :   jb_n3 : out std_logic; --PWM2 of first leg
119 :   jc_p0 : in std_logic; --A1 of second leg
120 :   jc_n0 : in std_logic; --B1 of second leg
121 :   jc_p1 : in std_logic; --A2 of second leg
122 :   jc_n1 : in std_logic; --B2 of second leg
123 :   jc_p2 : out std_logic; --DIR1 of second leg
124 :   jc_n2 : out std_logic; --PWM1 of second leg
125 :   jc_p3 : out std_logic; --DIR2 of second leg
126 :   jc_n3 : out std_logic; --PWM2 of second leg
127 :   jd_p0 : in std_logic; --A1 of third leg
128 :   jd_n0 : in std_logic; --B1 of third leg
129 :   jd_p1 : in std_logic; --A2 of third leg
130 :   jd_n1 : in std_logic; --B2 of third leg
131 :   jd_p2 : out std_logic; --DIR1 of third leg
132 :   jd_n2 : out std_logic; --PWM1 of third leg
133 :   jd_p3 : out std_logic; --DIR2 of third leg
134 :   jd_n3 : out std_logic; --PWM2 of third leg
135 :   je_p0 : in std_logic; --A1 of fourth leg
136 :   je_n0 : in std_logic; --B1 of fourth leg
137 :   je_p1 : in std_logic; --A2 of fourth leg
138 :   je_n1 : in std_logic; --B2 of fourth leg
139 :   je_p2 : out std_logic; --DIR1 of fourth leg
140 :   je_n2 : out std_logic; --PWM1 of fourth leg
141 :   je_p3 : out std_logic; --DIR2 of fourth leg
142 :   je_n3 : out std_logic; --PWM2 of fourth leg
143 :   led : out std_logic_vector(3 downto 0);
144 :   sw : in std_logic_vector(3 downto 0);
145 :   S_AXI_ACLK : in std_logic;

```

Figure 3.40: Top Module's Component Declaration - Ports.

Lastly, as with every component, the generic and port map fields must be completed, where we set the values of the component's generic variables (Fig.3.41) and ports (Fig.3.42).



```

labeled_labeled_ip_v1_0.vhd
/home/chris/ip_repo/labeled_labeled_ip_1.0/hdl/labeled_labeled_ip_v1_0.vhd

171 : -- Instantiation of Axi Bus Interface S00_AXI
172     laelaps_four_legs_duplicate_ip_v1_0_S00_AXI_inst : laelaps_four_legs_duplicate_ip_v1_0_S00_AXI
173         generic map (
174             sys_clk => sys_clk,           --system clock frequency in Hz
175             pwm_frequency => pwm_frequency, --PWM switching frequency in Hz
176             bits_resolution => bits_resolution, --bits of resolution setting the duty cycle
177             phases => phases,
178             bits => bits,
179             hip_ratio_numerator => hip_ratio_numerator,
180             hip_ratio_divisor => hip_ratio_divisor,
181             knee_ratio_numerator => knee_ratio_numerator,
182             knee_ratio_divisor => knee_ratio_divisor,
183             C_S_AXI_DATA_WIDTH => C_S00_AXI_DATA_WIDTH,
184             C_S_AXI_ADDR_WIDTH => C_S00_AXI_ADDR_WIDTH
185         )
186         port map (

```

Figure 3.41: Top Module's Component Generic Map.



```

labeled_labeled_ip_v1_0.vhd
/home/chris/ip_repo/labeled_labeled_ip_1.0/hdl/labeled_labeled_ip_v1_0.vhd

186 : port map (
187     jb_p0 => jb_p0,
188     jb_n0 => jb_n0,
189     jb_p1 => jb_p1,
190     jb_n1 => jb_n1,
191     jb_p2 => jb_p2,
192     jb_n2 => jb_n2,
193     jb_n3 => jb_n3,
194     jb_p3 => jb_p3,
195     jc_p0 => jc_p0,
196     jc_n0 => jc_n0,
197     jc_p1 => jc_p1,
198     jc_n1 => jc_n1,
199     jc_p2 => jc_p2,
200     jc_n2 => jc_n2,
201     jc_n3 => jc_n3,
202     jc_p3 => jc_p3,
203     jd_p0 => jd_p0,
204     jd_n0 => jd_n0,
205     jd_p1 => jd_p1,
206     jd_n1 => jd_n1,
207     jd_p2 => jd_p2,
208     jd_n2 => jd_n2,
209     jd_n3 => jd_n3,
210     jd_p3 => jd_p3,
211     je_p0 => je_p0,
212     je_n0 => je_n0,
213     je_p1 => je_p1,
214     je_n1 => je_n1,
215     je_p2 => je_p2,
216     je_n2 => je_n2,
217     je_n3 => je_n3,
218     je_p3 => je_p3,
219     led => led,
220     sw => sw,

```

Figure 3.42: Top Module's Component Port Map.

With this, all modifications are now complete and we are ready to package our IP.

### 3.3.3 Packaging the IP core

Now click on *Package IP* in the Flow Navigator (Under the PROJECT MANAGER tab) and you should see the Package IP tab (Fig.3.43). First we click in the *File Groups* tab, where we select the **Merge changes from File Groups Wizard** option (Fig.3.44).



Figure 3.43: Click on Package IP.



Figure 3.44: Merge changes from File Groups Wizard.

The same procedure is repeated for the *Customization Parameters*, *Ports and Interfaces* and *Customization GUI* tabs, where we select the corresponding **Merge changes from ... Wizard** option.

Finally, inside the *Review and Package* tab, we click the **Re-Package IP** option, which will finalize our Custom IP.

In the end, if warning messages appear inside the *File Groups* tab, relevant with the directory of the source files (Fig.3.45), then you can follow some steps mentioned later to correct this issue.



Figure 3.45: Warning Messages.

We close the *edit\_laelaps\_four\_legs\_duplicate\_ip\_v1\_0* project and return back to the original *labeled\_four\_legs\_tutorial* project. Inside the **Flow Navigator**, under the **IP INTEGRATOR** category, we select the **Create Block Design** option. This will open a Block Design Diagram, where we can add several IPs and useful blocks (Fig.3.46).



Figure 3.46: Block Design.

By default, numerous blocks and IPs are available, but sometimes the ones we have created are not visible. For this reason, we will add them to the current library. Again, through the **Flow Navigator**, under **PROJECT MANAGER**, we select **Settings** and then once the new window open, we click in the **Repository** option under the **IP** category (Fig.3.47). If our custom made IP (*laelaps\_four\_legs\_duplicate\_ip\_1.0 in this case*) is not visible, then we + button and browse in our file system to locate the *ip\_repo* folder or the folder where our custom IP is located (Fig.3.48). Once we click *Apply* and *OK*, our library will be updated and hence be able to find our custom IP.



Figure 3.47: Settings.



Figure 3.48: Ip\_repo inclusion.

To ensure that, we return to our Block Design and click the **Add IP** button (Fig.3.49). A search engine will appear, where we can search by name the IP or block we wish to include in our Block Design. After searching and selecting our Custom IP, we should be able to see its architecture and physical ports (Fig.3.50).



Figure 3.49: Add IP option.



Figure 3.50: Custom IP architecture and physical ports.

Now, if during the creation of the IP, some warnings appeared relevant to the directories of the source codes, you can follow the following steps. First, right click the Custom IP in the Block Design and select **Edit in IP Packager** option. This will open a new project, where modification can take place on our Custom IP (Fig.3.51).

We return to the *Sources* pane and select and delete the *qei* and *pwm\_freq* components of the IP, along with their internal components. In the next window, we select the "Also delete project local files/directories from disk" option and click "OK" (Fig.3.52).



Figure 3.51: Custom IP Project.



Figure 3.52: Delete the IP's components.

After the deletion of the component source files, the following state should be visible (Fig.3.53), which will be corrected by clicking the **Add Sources** "+" symbol. There, we select the "Add or create design sources" option and click Next, where we have the option to browse inside our folders and specifically inside the *src* folder, inside the *laelaps\_four\_legs\_duplicate\_ip\_1.0* folder. In case that folder does not exist, we create it and inside it we copy all the source files we created from their original location. Once we have selected all source files, we click "OK" and proceed with the "Finish" button (Fig.3.54).



Figure 3.53: State after the deletion.



Figure 3.54: Selection of the new source files.

After that we left click the **component.xml** file inside the *Source Pane* (Fig.3.55), which will open the **Package IP Manager**. Once again, we click the **Merge changer from File Groups Wizard** option and then all warnings should disappear.

Finally, inside the *Review and Package* tab, we click the **Re-Package IP** option, which will finalize our Custom IP (Fig.3.56). It is strongly recommended to create an archive of the IP, in the *After Packaging* section, which will create a corresponding zip file of our IP that can be easily transferred to other devices and computers.



Figure 3.55: Left click component.xml



Figure 3.56: Re-Package IP.

### 3.3.4 Configuring the Block Design

Since our Custom IP has been completed, we are now ready to insert the remaining blocks of our implementation. Initially, we include the *ZYNQ7 Processing System*, which represents the ARM Processor of the Zybo Development Board (Fig.3.57). On the top of the Block Design Diagram, the Designer Assistance will propose two options of action, from which we select to execute the **Run Block Automation** (Fig.3.58). A new window will pop up, which explains the details of this actions. We do not change the default options and select "OK" (Fig.3.59). Following that, we select the **Run Connection Automation** option and once again proceed with the default settings (Fig.3.63). Eventually, we result with the the architecture depicted in Fig.3.61. As you can see, the *Processor System Reset* and *AXI Interconnect* blocks have been automatically created, while several connections have been made. These settings are vital to our implementation as they ensure the valid functionality of the components and their communication via the AXI Interconnect.



Figure 3.57: ZYNQ7 Processing System.



Figure 3.58: Run Block Automation.



Figure 3.59: Details of Run Block Automation.



Figure 3.60: Details of Run Connection Automation.

The only thing that remains is to connect our Custom IP's inputs and outputs to their respective external ports. This is achieved by right clicking our IP and selecting the *Make External* option (Fig.3.62).



Figure 3.61: Resulting Architecture.



Figure 3.62: Make External Option.

Finally, we end up with the finalized architecture of our implementation, whose Layout and Routing can be optimized with the following shortcut buttons (Fig.3.64 and Fig.3.65).



Figure 3.63: Details of Run Connection Automation.



Figure 3.64: Zoom Fit Shortcut.



Figure 3.65: Regenerate Layout.

### 3.3.5 Constraints

As we had mentioned earlier, the names of the external and physical ports are configured in the constraints file *Zybo-Master.xdc*, which can be accessed through the *Sources* Pane. In that file, all physical ports of the Zybo Development Board are included and we simply have to uncomment the code lines, which correspond to the ports we wish to utilize. In this case, the switches, the LEDs and the Pmod Ports. As you can observe, for every port inside the **get\_ports** field, a special codename exists, through which we can refer to that port. As a result, it is important to ensure that these codenames match with the ones inside the Block Design Diagram (Fig.3.66, 3.67).

```

Zybo-Master.xdc
/home/chris/laelaps_four_legsTutorial/laelaps_four_legsTutorial.srsc/constrs_1/imports/src/Zybo-Master.xdc

11;
12; ##Switches
13: set_property -dict { PACKAGE_PIN G15 IOSTANDARD LVCMS33 } [get_ports { sw_0[0] }]; #IO_L19N_T3_VREF_35 Sch-SW0
14: set_property -dict { PACKAGE_PIN P15 IOSTANDARD LVCMS33 } [get_ports { sw_0[1] }]; #IO_L24P_T3_34 Sch-SW1
15: set_property -dict { PACKAGE_PIN W13 IOSTANDARD LVCMS33 } [get_ports { sw_0[2] }]; #IO_L4N_T0_34 Sch-SW2
16: set_property -dict { PACKAGE_PIN T16 IOSTANDARD LVCMS33 } [get_ports { sw_0[3] }]; #IO_19P_T1_D0S_34 Sch-SW3
17;
18;
19; ##Buttons
20: set_property -dict { PACKAGE_PIN R18 IOSTANDARD LVCMS33 } [get_ports { btn[0] }]; #IO_L28N_T3_34 Sch-BTN0
21: set_property -dict { PACKAGE_PIN P16 IOSTANDARD LVCMS33 } [get_ports { btn[1] }]; #IO_L24N_T3_34 Sch-BTN1
22: set_property -dict { PACKAGE_PIN V16 IOSTANDARD LVCMS33 } [get_ports { btn[2] }]; #IO_L18P_T2_34 Sch-BTN2
23: set_property -dict { PACKAGE_PIN Y16 IOSTANDARD LVCMS33 } [get_ports { btn[3] }]; #IO_L7P_T1_34 Sch-BTN3
24;
25;
26; ##LEDs
27: set_property -dict { PACKAGE_PIN M14 IOSTANDARD LVCMS33 } [get_ports { led_0[0] }]; #IO_L23P_T3_35 Sch-LED0
28: set_property -dict { PACKAGE_PIN M15 IOSTANDARD LVCMS33 } [get_ports { led_0[1] }]; #IO_L23N_T3_35 Sch-LED1
29: set_property -dict { PACKAGE_PIN G14 IOSTANDARD LVCMS33 } [get_ports { led_0[2] }]; #IO_0_35-Sch-LED2
30: set_property -dict { PACKAGE_PIN D18 IOSTANDARD LVCMS33 } [get_ports { led_0[3] }]; #IO_L3N_T0_D0S_ADIN_35 Sch-LED3
31;

```

Figure 3.66: Code Lines of Constraints File relevant with the Switches and LEDs.

```

Zybo-Master.xdc
/home/chris/laelaps_four_legsTutorial/laelaps_four_legsTutorial.srsc/constrs_1/imports/src/Zybo-Master.xdc

80; ##Pmod Header J0
81: set_property -dict { PACKAGE_PIN T20 IOSTANDARD LVCMS33 } [get_ports { jb_p[0] }]; #IO_L15P_T2_D0S_34 Sch-J01_P
82: set_property -dict { PACKAGE_PIN L20 IOSTANDARD LVCMS33 } [get_ports { jb_n[0] }]; #IO_L15N_T2_D0S_34 Sch-J01_N
83: set_property -dict { PACKAGE_PIN V20 IOSTANDARD LVCMS33 } [get_ports { jb_p[1] }]; #IO_L16P_T2_34 Sch-J02_P
84: set_property -dict { PACKAGE_PIN W20 IOSTANDARD LVCMS33 } [get_ports { jb_n[1] }]; #IO_L16N_T2_34 Sch-J02_N
85: set_property -dict { PACKAGE_PIN Y18 IOSTANDARD LVCMS33 } [get_ports { jb_p[2] }]; #IO_L17P_T2_34 Sch-J03_P
86: set_property -dict { PACKAGE_PIN X18 IOSTANDARD LVCMS33 } [get_ports { jb_n[2] }]; #IO_L17N_T2_34 Sch-J03_N
87: set_property -dict { PACKAGE_PIN W18 IOSTANDARD LVCMS33 } [get_ports { jb_p[3] }]; #IO_L22P_T2_34 Sch-J04_P
88: set_property -dict { PACKAGE_PIN V19 IOSTANDARD LVCMS33 } [get_ports { jb_n[3] }]; #IO_L22N_T2_34 Sch-J04_N
89;
90;
91; ##Pmod Header J0
92: set_property -dict { PACKAGE_PIN V15 IOSTANDARD LVCMS33 } [get_ports { jc_p[0] }]; #IO_L18P_T1_34 Sch-JC1_P
93: set_property -dict { PACKAGE_PIN W15 IOSTANDARD LVCMS33 } [get_ports { jc_n[0] }]; #IO_L18N_T1_34 Sch-JC1_N
94: set_property -dict { PACKAGE_PIN V16 IOSTANDARD LVCMS33 } [get_ports { jc_p[1] }]; #IO_L17P_T0_34 Sch-JC2_P
95: set_property -dict { PACKAGE_PIN W16 IOSTANDARD LVCMS33 } [get_ports { jc_n[1] }]; #IO_L17N_T0_34 Sch-JC2_N
96: set_property -dict { PACKAGE_PIN V14 IOSTANDARD LVCMS33 } [get_ports { jc_p[2] }]; #IO_L2P_T1_34 Sch-JC3_P
97: set_property -dict { PACKAGE_PIN W14 IOSTANDARD LVCMS33 } [get_ports { jc_n[2] }]; #IO_L8N_T1_34 Sch-JC3_N
98: set_property -dict { PACKAGE_PIN V12 IOSTANDARD LVCMS33 } [get_ports { jc_p[3] }]; #IO_L2P_T0_34 Sch-JC4_P
99: set_property -dict { PACKAGE_PIN W12 IOSTANDARD LVCMS33 } [get_ports { jc_n[3] }]; #IO_L2N_T0_34 Sch-JC4_N
100;

Zybo-Master.xdc
/home/chris/laelaps_four_legsTutorial/laelaps_four_legsTutorial.srsc/constrs_1/imports/src/Zybo-Master.xdc

102; ##Pmod Header J0
103: set_property -dict { PACKAGE_PIN T14 IOSTANDARD LVCMS33 } [get_ports { jd_p[0] }]; #IO_L5P_T0_34 Sch-J01_P
104: set_property -dict { PACKAGE_PIN T15 IOSTANDARD LVCMS33 } [get_ports { jd_n[0] }]; #IO_L5N_T0_34 Sch-J01_N
105: set_property -dict { PACKAGE_PIN V14 IOSTANDARD LVCMS33 } [get_ports { jd_p[1] }]; #IO_L6P_T0_34 Sch-J02_P
106: set_property -dict { PACKAGE_PIN W14 IOSTANDARD LVCMS33 } [get_ports { jd_n[1] }]; #IO_L6N_T0_VREF_34 Sch-J02_N
107: set_property -dict { PACKAGE_PIN V13 IOSTANDARD LVCMS33 } [get_ports { jd_p[2] }]; #IO_L11P_T1_34 Sch-J03_P
108: set_property -dict { PACKAGE_PIN W13 IOSTANDARD LVCMS33 } [get_ports { jd_n[2] }]; #IO_L11N_T1_SRCC_34 Sch-J03_N
109: set_property -dict { PACKAGE_PIN V17 IOSTANDARD LVCMS33 } [get_ports { jd_p[3] }]; #IO_L21P_T3_D0S_34 Sch-J04_P
110: set_property -dict { PACKAGE_PIN W18 IOSTANDARD LVCMS33 } [get_ports { jd_n[3] }]; #IO_L21N_T3_D0S_34 Sch-J04_N
111;
112;
113; ##Pmod Header J0
114: set_property -dict { PACKAGE_PIN V12 IOSTANDARD LVCMS33 } [get_ports { je_p[0] }]; #IO_L4P_T0_34 Sch-JE1
115: set_property -dict { PACKAGE_PIN N16 IOSTANDARD LVCMS33 } [get_ports { je_n[0] }]; #IO_L18P_T2_34 Sch-JE2
116: set_property -dict { PACKAGE_PIN V15 IOSTANDARD LVCMS33 } [get_ports { je_p[1] }]; #IO_25_35 Sch-JE3
117: set_property -dict { PACKAGE_PIN H15 IOSTANDARD LVCMS33 } [get_ports { je_n[1] }]; #IO_L19P_T3_35 Sch-JE7
118: set_property -dict { PACKAGE_PIN V13 IOSTANDARD LVCMS33 } [get_ports { je_p[2] }]; #IO_L3N_T0_D0S_34 Sch-JE8
119: set_property -dict { PACKAGE_PIN U17 IOSTANDARD LVCMS33 } [get_ports { je_n[2] }]; #IO_L2N_T1_D0S_34 Sch-JE8
120: set_property -dict { PACKAGE_PIN V17 IOSTANDARD LVCMS33 } [get_ports { je_p[3] }]; #IO_L26P_T3_34 Sch-JE9
121: set_property -dict { PACKAGE_PIN Y17 IOSTANDARD LVCMS33 } [get_ports { je_n[3] }]; #IO_L7N_T1_34 Sch-JE10
122;

```

Figure 3.67: Code Lines of Constraints File relevant with the Pmod Ports.

### 3.3.6 Configuring the Operating Frequency of the FPGA

A critical aspect of our hardware design is its operating frequency. Specifically, the FPGA's operating frequency determines the period of the overall's system's clock signal.

One way to configure this frequency is through the **Block Design Diagram**, where we double click the *ZYNQ7 Processing System* Block. A **Re-customize IP** window will open, with a *Clock Configuration* section visible, under the the *Page Navigator* category (Fig.3.68). In fact, the system's clock is the *FCLK\_CLK0* and it is located under the *PL Fabric Clocks*. By modifying the Requested Frequency entry, we can determine the system's frequency, which can receive selected values between 0.1MHz and 250MHz. The users do not have to know these specific values, but they can simply insert their desired value and the system will find the nearest available value, which will be visible in the *Actual Frequency* entry.

We should note that choosing the maximum available frequency is not always the optimal solution. First of all, sometimes our designs cannot support all available values. In those cases, our system cannot respond to the given frequency and therefore a lower frequency should be adopted. On the other hand, sometimes our system can operate at higher frequencies than the one we have selected. However, although selecting a higher value would accelerate our processing speed, it would also increase the system's power consumption, a feature that is not always preferable.

A useful indicator for the designer is the *Worst Negative Slack (WNS)* and *Total Negative Slack (TNS)* metrics, which are created alongside with the Timing Report of the Project's Summary. This report is generated automatically during the generation of the Bitstream file, whose process will be explained in the next paragraph. If those metrics have negative values, then this indicates that our system cannot support the current frequency and hence a lower value should be selected, before re-generating the Bitstream File. Conversely, if they have positive values, the operating frequency can be increased.

For the laelaps\_four\_legs\_tutorial, the maximum available frequency was found to be around 83 MHz (83.333336 MHz) and hence we selected this value, before generating the Bitstream File.



Figure 3.68: Configuration of the FPGA's Operating Frequency.

### 3.3.7 Generate Bitstream File

The last part of the Hardware Design of our implementation is to create a Bitstream file. To do that, we have to first create the HDL Wrapper of our design, by right clicking our design inside the *Design Sources* category of the *Sources* Pane (Fig.3.69). After we select the **Vivado manage wrapper and auto-update** option, we are ready to Generate our Bitstream, via the appropriate command in **Flow Navigator** pane, under the **PROGRAM AND DEBUG** category. Since there are no implementation results available, we hit "OK", in the pop up window to launch synthesis and implementation as well. In the **Launch Runs** window, we choose to *Launch runs on local host*, where the *Number of jobs*, corresponds to the number of processors we want to employ for this action. Due to the fact that the generation of the Bitstream usually takes around 15 to 30 minutes, we recommend to utilize as many processors as possible. You can observe the procedure of the generation, by clicking on the **PROJECT MANAGER** under the **Flow Navigator** Pane. In Fig.3.70 the Project Summary is observed, during the generation of the Bitstream File.



Figure 3.69: Create HDL Wrapper.



Figure 3.70: Project Summary.

Once the generation has been concluded, the Project summary informs us about various important aspects of our design (Fig.3.71). First of all, the Status of the **Synthesis** and **Implementation** sections, reassures us that no syntax errors exist in our implementation. Secondly, any critical and normal warnings notify us regarding some characteristics of our design that may prove to be troublesome or not. In Fig.3.71 the critical warning is not relevant with our design and therefore we will not delve into it. The **Timing Report** shows the values of the slack metrics, which in the 83 MHz Project Summary are nonnegative and rather small. This indicates that the given frequency is suitable, while in the 100 MHz Project Summary (Fig.3.72) the slack metrics are negative and hence indicate the necessity to decrease the frequency. The **Utilization Report** depicts the overall resource consumption of the project, which is elaborated in the Diploma Thesis <sup>¶</sup>. Lastly, the Power Report shows estimates regarding the On-chip Power and Temperature, although they are not accurate and therefore we do not rely on them.



Figure 3.71: Finalized Project Summary.



Figure 3.72: Project Summary with Negative Slack for 100 MHz Operating Frequency.

For more information, the following Programmable Logic Tutorials are strongly recommended: Getting Started with Zynq, Creating IP in HDL and Creating a Custom IP core using the IP Integrator.

<sup>¶</sup>Page 64 in Chapter 4.

# Chapter 4

## Software Partition in SDK

As soon as the hardware development of our implementation on Vivado has been concluded, we can initiate the software development that will be executed to the ARM Processor of the Zynq Development Board. For this purpose an Integrated Design Environment for creating embedded applications is provided by Xilinx, the Xilinx Software Development Kit (XSDK).

### 4.1 Export the hardware design to SDK

After the generation of the Bitstream File, we export our design to SDK, in order to link the software and hardware partitions. In Vivado, from the File menu, select *Export → Export Hardware* (Fig.4.1). In the window that appears, tick *Include bitstream* and click *OK* (Fig.4.2). Again from the File menu, select *Launch SDK*. In the window that appears, use the settings of Fig.4.3 and click *OK*.



Figure 4.1: Export Hardware.



Figure 4.2: Export Hardware - Include Bitstream.



Figure 4.3: Launch SDK Settings.

At this point, the SDK loads and a hardware platform specification will be created for your design. You should be able to see the hardware specification in the Project Explorer of SDK as shown in the image below (Fig.4.4). You are now ready to create a software application to run on the PS.



Figure 4.4: SDK Home Screen.

## 4.2 Preliminary Settings for the SDK

From the File menu, select *New → Application Project*. A New Project window will appear, through which we will create a FSBL project file, which is required in order to make our implementation bootable from a SD Card (Fig.4.5). Once a name has been chosen for the project, *laelaps\_four\_legs\_tutorial\_fsbl* in this case, we click *Next*. In the **Templates** dialog box, select the *Zynq-7000 AP SoC FSBL* template (Fig.4.6) and then click *Finish*. As a result, the application project will be created inside the *Project Explorer* pane, along with the corresponding board support package (bsp file). For more information, the following Tutorial is strongly recommended: Creating a New Zynq FSBL Application Project .



Figure 4.6: Template for Zynq FSBL.

Figure 4.5: FSBL Project File.

Consequently, again from the File menu, select *New → Application Project*. A New Project window will appear, through which we will create our software application file that will be responsible for the programming of the ARM Processor (Fig.4.7). Once a name has been chosen for the project, *laelaps\_four\_legs\_tutorial* in this case, we click *Next*. In the **Templates** dialog box, select the *Hello World* template (Fig.4.8) and then click *Finish*. As a result, the software application project will be created inside the *Project Explorer* pane, along with the corresponding board support package (bsp file).



Figure 4.7: Software Application Project File.

Now, from the Project Explorer pane we navigate inside the *laelaps\_four\_legs\_tutorial* folder and open the **helloworld.c** file, by double clicking it (Fig.4.9). This is the source file that will be modified in order to execute our implementation.



Figure 4.9: The *laelaps\_four\_legs\_tutorial* folder.

## 4.3 Modify the Software Application

By default, the **helloworld.c** source file is a basic application that simply prints a "Hello World" message (Fig.4.10). Based on this example, the user can experiment and eventually formulate his/her desired application in C or C++ language.



```
/*
 * helloworld.c: simple test application
 *
 * This application configures UART 16550 to baud rate 9600.
 * PS7 UART (Zynq) is not initialized by this application, since
 * bootrom/bsp configures it to baud rate 115200
 *
 * -----
 * | UART TYPE   BAUD RATE
 * -----
 * | uartns550  9600
 * | uartlite   Configurable only in HW design
 * | ps7_uart   115200 (configured by bootrom/bsp)
 */
#include <stdio.h>
#include "platform.h"
#include "xil_printf.h"

int main()
{
    init_platform();
    print("Hello World\n\r");
    cleanup_platform();
    return 0;
}
```

Figure 4.10: The helloworld.c initial code.

As a remainder, in our implementation the processor is responsible for the execution of both the High and Low level controllers, as well as the retention of the necessary data for post processing using Matlab. However, although the software segment of the implementation has been explained in the context of the Diploma Thesis \*, some parts have been altered and hence it is required to analyze them.

First of all, an Interrupt Service Routine (ISR) has been included, which provides increased temporal precision and enables the interfacing with the user via a terminal. Specifically, we can determine how frequently an interrupt occurs, which causes the execution of the Low-level controller that has been placed inside the corresponding interrupt handler. On the other hand, the main function emulates the High-level controller, while handling the interfacing with the user. Finally, after the duration of the locomotion experiment has finished, all recorded data are transferred to the same SD card, where our implementation boots from.

For the installation of the ISR, we followed the examples described in the **Zynq Workshop for Beginners** and especially the 8th exercise, whose solution is available here. The complete pdf file, along with useful code templates and solutions to its exercises are available here, while the pdf and the 8th exercise's code are also included inside the "diploma\_thesis\_code.zip" file.

### 4.3.1 Main Function

Based on the 8th exercise solution of the **Zynq Workshop for Beginners**, we only modified certain parts to introduce the desired performance. Once all necessary declarations have been made, an infinite while loop initiates, where the interrupts are enabled and the frequency of the interrupts is configured (Fig.4.11). Currently the frequency has been set to 1kHz, since it has been proven suitable for our application in the experiments so far.

\*Chapter 5 of the Diploma Thesis.

```

// Load the timer with a value that represents one second of real time
// HINT: The SCU Timer is clocked at half the frequency of the CPU.
XScuTimer_LoadTimer(&my_Timer, XPAR_PS7_CORTEXA9_0_CPU_CLK_FREQ_HZ*pow(10,-3) / 2);
//XPAR_PS7_CORTEXA9_0_CPU_CLK_FREQ_HZ / 2 -> 1Hz
//XPAR_PS7_CORTEXA9_0_CPU_CLK_FREQ_HZ*pow(10,-3) / 2 -> 1kHz

```

Figure 4.11: Interrupt Frequency Configuration.

The *XPAR\_PS7\_CORTEXA9\_0\_CPU\_CLK\_FREQ\_HZ* constant represents the frequency of the CPU that is equal to 650MHz. Since the SCU Timer is clocked at half that frequency, if we load the *XPAR\_PS7\_CORTEXA9\_0\_CPU\_CLK\_FREQ\_HZ / 2* value to the timer, every second (1Hz frequency) an interrupt will occur. According to that logic, we configured the frequency to 1kHz.

Next, the interface with the user begins, by printing a welcome message and a request for the state machine code. Inside a while loop, which checks whether a flag variable is equal to '0', the system continuously awaits for a user input from the terminal via a non-blocking function. When the code "1" is given, an interrupt counter is reset, according to which we can measure the number of interrupts that have occurred and hence the time that has intervened since the pressing of the "1" code. Simultaneously, a counter variable is also reset to zero, which designates where the next experiment measurement should be temporarily stored inside the global *array\_print* array. Lastly, the flag variable is set to '1' and another message is printed to the terminal, informing the user that the trotting experiment has started.

Following, the system awaits again inside a while loop, which checks whether the flag variable is equal to '1', for a user input from the terminal via a non-blocking function. In case the user types "0", the experiment is terminated, the flag variable is set to '2', and a corresponding message is printed. Otherwise, the while loop continues.

Afterwards, a series of if statements take place that help to deduce the current state of the flag variable. If the flag variable is equal to '2', it indicates that the user commanded the termination of the program and hence the control gains are set to zero. Furthermore, a delay of 0.5 sec is inserted to ensure that at least one more interrupt will occur, which will set all duty cycles to zero, in order to disable the PWM signals sent to the motors. Moreover, the system breaks from the infinite while loop and once all interrupts have been disabled, the main function exits. If the flag variable is equal to '3', it indicates that the experiment was terminated due to the expiration of the experiment's duration and hence the storing of the recorded data begins. A corresponding message is printed, the interrupts are disabled and the *store\_data\_in\_sd* function is called. This function reads and prints to the SD Card, all data stored inside the global *array\_print* array, where all the experiment measurements are temporarily saved. Once the printing has concluded (usually takes around 20 seconds), the system exits.

### 4.3.2 *array\_print* function

This function was created based on the examples found here and here. In order to utilize the *f\_printf* function to print our data to the SD Card file, certain adjustments are required for the SDK program. Inside Xilinx SDK, open your Board Support Package (system.mss file) and select Modify this BSP's Settings (Fig.4.12). Enable the xilffs library and then under Overview, you can select xilffs (Fig.4.13). As instructed here, in order for the *f\_printf* function to be available *read\_only* must be equal to 0, while *use\_strfunc* must be equal to either 1 or 2.(Fig.4.14)



Figure 4.12: Modify this BSP's Settings



Figure 4.13: Enable xilffs library.



Figure 4.14: Configuration for library xilffs.

### 4.3.3 Interrupt Handler

Every time an interrupt happens, the program control transits to the *my\_timer\_interrupt\_handler* function, which represents its interrupt handler. After the declaration and definition of all necessary variables for the Low-level controller, certain default commands pertinent with the functionality of the interrupt are included, while the global interrupt counter is increased. Based on that counter, we can measure time, given that every 1ms an interrupt takes place. For example, if the interrupt counter is equal to 5000, then 5 seconds have intervened since the reset of the interrupt counter, which happens after the user has pressed "1" in the main function. Therefore, we can keep track of time since the beginning of the trotting experiment and later deduce whether it has exceeded the desired duration.

Subsequently, several if statements help us find out at which state our system is. If the global flag variable *flag* is equal to '0' and we have not exceeded the first *lerp\_t\_interval* seconds of the experiment, then the user has not yet pushed the "1" button and hence we simply execute control to zero point. The ellipse parameters are set to zero, while the control gains steadily reach their final values through a linear interpolation of *lerp\_t\_interval* seconds.

If the *flag* is equal to '1' and the first *lerp\_t\_interval* seconds of the experiment have not passed, the ellipse parameters start to increase progressively to their final values through linear interpolation of *lerp\_t\_interval* seconds.

If the *flag* is equal to '1' and less than *lerp\_t\_interval* seconds remain for the experiment to finish, the ellipse parameters start to decrease progressively to their initial values through linear interpolation of *lerp\_t\_interval* seconds.

If the *flag* is equal to '2', than the user has ordered the termination of the experiment and the control gains are set to zero, in order to disable the PWM outputs to the motors.

Ultimately, if the duration of the experiment has exceeded its defined limit and the *flag* variable is '1', then the *flag* is set to '3' and the Control gains are set to zero, in order to stop the trotting experiment and initiate the storage of the recorded data.

Next, the Low-level controller is executed, utilizing the parameters that were defined inside the if statements previously. As described in the 5th chapter of the Diploma Thesis, first the inverse kinematics and the trajectory planning are executed, and afterwards the PD-Controller is applied. For the communication with the FPGA, the following pointer is utilized, which is linked with the memory address of the first slave register of the AXI4-Lite Interface (Fig.4.15). The memory addresses dedicated to the AXI4-Lite Interface are specified in the system.hdl file of our project (Fig.4.16).

```
//Pointer to the Memory Address of the first slave register
Xuint32 *baseaddr_p = (Xuint32 *) XPAR_LAELEPS_FOUR_LEGS_DUPLICATE_IP0_S00_AXI_BASEADDR;
//*(baseaddr_p+1) refers to the second slave register and so on
```

Figure 4.15: Pointer to the first slave register's memory address.

| Cell                             | Base Addr  | High Addr  | Slave I/F | Mem/Reg  |
|----------------------------------|------------|------------|-----------|----------|
| laelaps_four_legs_duplicate_ip_0 | 0x43c00000 | 0x43cfffff | S00_AXI   | REGISTER |
| ps7_af1_0                        | 0xf8008000 | 0xf8008fff |           | REGISTER |
| ps7_af1_1                        | 0xf8009000 | 0xf8009fff |           | REGISTER |

Figure 4.16: Memory Address specified in the system.hdf file.

## 4.4 Libraries Installation

Our final source file utilizes a lot of mathematical functions that are included normally in the "math.h" C library. In order to have access to that library, we have to carry out the following steps.

From the Project menu, select Properties (Fig.4.17). In the next window, under *C/C++ Build* click on *Settings* and then under *ARM v7 gcc linker* click on *Libraries*. Afterwards, click on the Add button (Fig.4.19), which allows us to include the "m" library we want (Fig.4.18). After this, there should no problem using any mathematical functions.



Figure 4.18: Add m Library.

Figure 4.17: Project → Properties.



Figure 4.19: Add Library.

## 4.5 Load executables to the SD Card

When the editing of the software partition has also been completed, we can transfer our source files to the SD Card that will later be mounted into the Zybo Development Board. Specifically, a BOOT.bin will be created, which will entail both the software and hardware segments of our implementation. The programming of the device can also be realized through a USB cable, however this method was proved to be defective.

First, we simply select the *laelaps\_four\_legs\_tutorial\_fsbl* file inside the **Project Explorer** pane, and then from the *Xilinx* menu at the top of the screen, we click the *Create Boot Image* option (Fig.4.20). A new window will appear with several default settings that do not require modification. The only thing that is necessary is to add the file path of our software application in the *Boot image partitions* section. Of course, the *Output path* can be modified to any convenient location for the user, as this is where the BOOT.bin file will be stored afterwards. Thankfully, as you can see, the Bitstream and fsbl files are already included and hence we simply have to click on the *Add* button and browse to the .elf file of our software application project, *laelaps\_four\_legs\_tutorial.elf* in this case (Fig.4.23). Ultimately, the following status should be visible (Fig.4.24), where all that remains is to click the *Create Image* button. If we now browse to the location specified in the *Output path*, we should find the BOOT.bin file, which afterwards has to be transferred to a FAT-32 formatted SD Card. On the same SD Card, a .txt file will be created after the execution of the implementation. The name of this file can be configured in the initial declaration of the C code (Fig.4.21). Naturally, if any modifications take place either on the hardware or software partition, the whole Create Boot Image procedure has to be repeated, and the new BOOT.bin file has to be retransferred to the SD Card.



Figure 4.20: Create Boot Image.

```
//the name of the .txt file, where the recorded data will be stored
static char FileName[32] = "0:/reoutput.txt";
```

Figure 4.21: Configuration of the .txt file's name.



Figure 4.22: Create Boot Image Settings.



Figure 4.23: Location of the Software Application Project .elf file.



Figure 4.24: Final status of the Create Boot Image window.

The SD Card containing the BOOT.bin file should be inserted in the *microSD connector (Reverse side)* (Port No.25 of Fig.4.25) and the *Programming Mode Jumper* (No.21 of Fig.4.25) should be set to the two pin headers on the left. Moreover, the device should be connected to a PC via a USB cable (Port No.3 of Fig.4.25). As soon as the Board is powered on, the execution of the source codes begins. If we want to reset the Processor, we can simply press the Processor Reset Pushbutton (Port No.15 of Fig.4.25), and for a complete reset, we can press the *Logic configuration reset Pushbutton* (Port No.16 of Fig.4.25).



Figure 4.25: ZYBO Device Diagram.

Additionally, the encoders' of the motors and the PWM Ethernet cables should be connected to the Secondary Boards. Each socket has been numbered with its respective code that is created in the software partition and hence the REOUTPUT.TXT file. As a result, the user can connect each motor's signals wherever he/she wants to, but has to invert the angles that correspond to right legs' signals (Refer to the last lines of the *my\_timer\_interrupt\_handler* function in the Software partition) and update the Matlab code for the visualization (Refer to the next chapter). As you can see, the .txt file entails useful information regarding each joint of the robot, recorded at several moments of time. In fact, each line consist of the respective joint's codename (1-8), the specific time instant of the recording (0.001 refers to 1ms), the angle (degrees) and velocity of the joint (rad/s), the Duty Cycle of the PWM signal that will be sent to the joint's motor (+/- for Positive/Negative Direction), and the desired angle of the joint (degrees).

| REOUTPUT.TXT (2,0 GB Volume /media/chris/0762-D941) - gedit |      |      |   |       |        |         |
|-------------------------------------------------------------|------|------|---|-------|--------|---------|
|                                                             | Open | Save |   |       |        |         |
| REOUTPUT.TXT                                                |      |      | 1 | 0.001 | -0.000 | -0.000  |
|                                                             |      |      | 2 | 0.001 | -0.000 | -0.000  |
|                                                             |      |      | 3 | 0.001 | -0.000 | -0.000  |
|                                                             |      |      | 4 | 0.001 | -0.000 | -0.000  |
|                                                             |      |      | 5 | 0.001 | -0.000 | -0.000  |
|                                                             |      |      | 6 | 0.001 | -0.000 | -0.000  |
|                                                             |      |      | 7 | 0.001 | +0.002 | -0.000  |
|                                                             |      |      | 8 | 0.001 | -0.000 | -0.000  |
|                                                             |      |      |   |       |        | -30     |
|                                                             |      |      |   |       |        | -21.647 |
|                                                             |      |      |   |       |        | +15.276 |
|                                                             |      |      |   |       |        | +21.647 |
|                                                             |      |      |   |       |        | -15.276 |
|                                                             |      |      |   |       |        | -21.647 |
|                                                             |      |      |   |       |        | +15.276 |
|                                                             |      |      |   |       |        | +21.647 |
|                                                             |      |      |   |       |        | -15.276 |

Figure 4.26: Form of the REOUTPUT.TXT file.



Figure 4.27: Sockets 3-4-5-6 of the Secondary Boards.



Figure 4.28: Sockets 1-2-7-8 of the Secondary Boards.

## Chapter 5

# Execution of the Implementation

As mentioned above, the device is connected with a PC through a USB Cable, which is required for the interfacing with the user. Specifically, the user can insert commands and observe any messages printed from the processor. This feature is achieved through serial communication via the serial port. In order to monitor that communication the Minicom program is utilized (Section 2.1). In Fig.5.1 we can observe the Start Menu of Minicom through which we can configure the settings of the serial communication.



Figure 5.1: Start Menu of Minicom.

As soon as the device has been powered on, we can observe messages printed out in the terminal, while the users can insert their commands when it is required, as programmed in the software application. In the following figures the output of the system is illustrated for two different executions: in both of them, the system is commanded by the user to initiate the trotting experiment and in the first one (Fig.5.2) it is terminated after the expiration of its duration, while in the second one (Fig.5.3), the user commands its termination.

Once the execution of the implementation has been concluded, we can remove the SD Card and insert in our PC. Inside it, apart from the BOOT.bin file, we should find a .txt file (REOUTPUT.TXT in our case Fig.4.21), which will entail all recorded measurements from the experiment (Fig.5.4). In order to visualize these data, a post-processing phase takes place in Matlab.



```
chris@chris-SATELLITE-L50-A-19P: ~
Welcome to minicom 2.7

OPTIONS: I18n
Compiled on Nov 15 2018, 20:18:47.
Port /dev/ttyUSB1, 14:58:29

Press CTRL-A Z for help on special keys

Start of the experiment! Provide state machine code:
State machine code is : 1. Initialization of Trotting.
Termination of the experiment! Initialization of data storage in SD Card.
The value of i is : 160000
Termination of the data storage in SD Card. Exiting!
```

Figure 5.2: Printed Messages After Execution - Termination due to expiration of experiment's duration.



```
chris@chris-SATELLITE-L50-A-19P: ~
Welcome to minicom 2.7

OPTIONS: I18n
Compiled on Nov 15 2018, 20:18:47.
Port /dev/ttyUSB1, 17:08:51

Press CTRL-A Z for help on special keys

Start of the experiment! Provide state machine code:
State machine code is : 1. Initialization of Trotting.
Experiment was terminated by user.
```

Figure 5.3: Printed Messages After Execution - Termination by user.



Figure 5.4: SD Card folder after execution.

# Chapter 6

## Matlab Codes

The .txt file that is created after the execution (REOUTPUT.TXT), is given as input to the *velocity\_estimation\_four\_legs\_inv\_kin.m* Matlab code, which creates four figures: the response of each leg's end effector in the steady state, the response of each joint's angle, the response of each joint's velocity, and the response of each joint's PWM Command. Through them, we can validate our experiment and observe its outcome.

According to where each motor signals were connected the Matlab code has to be updated, in order to ensure that the titles of the figures refer to the right joints. For example, in Fig.6.1 the lines with codenames 1 and 2 of the REOUTPUT.txt file correspond to the Hinder Right leg, the signals with codenames 3 and 4 of the .txt file correspond to the Hinder Left leg and so on. That way we can ensure that the produced figures represent the correct joints.

```
8 - A7=A(find(A(:,1)==1),2:6); %these are for the HR leg
9 - A8=A(find(A(:,1)==2),2:6);
10 - A5=A(find(A(:,1)==3),2:6); %these are for the HL leg
11 - A6=A(find(A(:,1)==4),2:6);
12 - A3=A(find(A(:,1)≠5),2:6); %these are for the FR leg
13 - A4=A(find(A(:,1)==6),2:6);
14 - A1=A(find(A(:,1)==7),2:6); %these are for the FL leg
15 - A2=A(find(A(:,1)==8),2:6);
```

Figure 6.1: Match-up of figures' titles and connected motor signals.

The *velocity\_estimation\_four\_legs\_inv\_kin.m* is available inside the *diploma\_thesis\_codes* folder as well.